python标准库 datetime的astimezone设置时区遇到的坑及解决

目录
  • datetime的astimezone设置时区遇到的坑
    • 1、datetime 获取当前的本地日期和UTC日期
    • 2、现在定义时区对象
    • 3、现在在 UTC 的时间基础上,用 astimezone() 修改时区
  • python2和python3的datetime时区问题:timezone时间转换
    • 1.python2.7
    • 2. python3

datetime的astimezone设置时区遇到的坑

datetime有四个主要的模块:

  • 1、date 处理年、月、日。
  • 2、time 处理时、分、秒和分数。
  • 3、datetime 处理日期和时间同时出现的情况。
  • 4、timedelta 处理日期和/ 或时间间隔。

1、datetime 获取当前的本地日期和UTC日期

# 从 datetime 中导入 datetime 模块。
from datetime import datetime
 
 
utc_time = datetime.utcnow()    # utcnow()获取世界标准时间。
print(f'1、UTC时间为:{utc_time}')
 
local_time = datetime.now()     # now() 获取本地时间。
print(f'1、本地时间为:{local_time}')

运行结果为:

1、UTC时间为:2019-06-26 10:58:24.730439

1、本地时间为:2019-06-26 18:58:24.730439

可以看出本地时间早于UTC世界标准时间8小时。

2、现在定义时区对象

利用 timezone() 可以设置时区对象,里面参数可由 和 timedelta() 提供。

timedelta() 是用来实现对日期的加减。

参数如下:

timedelta([days[, seconds[, microseconds[, milliseconds[, minutes[, hours[, weeks]]]]]]]) 

从构造函数的定义中可以看出,所有参数都是可选的,并且默认都是0。

from datetime import timezone, timedelta
 
beijing = timezone(timedelta(hours=8))
print(f'1、北京时区为:{beijing}')
 
Tokyo = timezone(timedelta(hours=9))
print(f'2、东京时区为:{Tokyo}')
 
New_York = timezone(timedelta(hours=-4))
print(f'3、纽约时区为:{New_York}')
 
utc = timezone.utc
print(f'4、世界标准时区为:{utc}')

运行结果为:

1、北京时区为:UTC+08:00

2、东京时区为:UTC+09:00

3、纽约时区为:UTC-04:00

4、世界标准时区为:UTC

可以看出定义的时区都是在 UTC时区 基础上进行的 加或减 操作。

3、现在在 UTC 的时间基础上,用 astimezone() 修改时区

1)下面是 错误示范:

utc_time = datetime.utcnow()
print(f'UTC时间为:{utc_time}')
print(f'本地时间为:{datetime.now()}')
 
time_beijing = utc_time.astimezone(beijing)
time_tokyo = utc_time.astimezone(Tokyo)
time_newyork = utc_time.astimezone(New_York)
 
print('1、更改时区为北京后的时间:', time_beijing)
print('2、更改时区为东京后的时间:', time_tokyo)
print('3、更改时区为纽约后的时间:', time_newyork)

运行结果如下:

UTC时间为:2019-06-26 11:55:18.150625
本地时间为:2019-06-26 19:55:18.150625
 
1、更改时区为北京后的时间: 2019-06-26 11:55:18.150625+08:00  # 后面的 +08:00 只是显示时区,
                                                            # 这里表示的时间就是 11 点,而不是加了 8 后的 19 点。
2、更改时区为东京后的时间: 2019-06-26 12:55:18.150625+09:00  # 东京时间早于北京时间1小时。
3、更改时区为纽约后的时间: 2019-06-25 23:55:18.150625-04:00  # 纽约时间晚于北京时间12小时。

可以看出运行结果的 北京时间 和 本地时间 差了 8 小时 和 UTC 时间 是一样的。说明这样的时区调整结果是不对的。

正常情况,北京时间 应该和 本地时间 一样。

# 用 tzinfo 查看 utc_time 时区属性

print(utc_time.tzinfo)

结果为:None。可知上面日期错误的原因是:

用 astimezone() 对 utc_time 的时区属性进行修改时,因为utc_time的时区属性为None,所以模块会把 utc_time 的时区默认为本地时区。

因为本地时区就是 UTC+08:00,所以当用  time_beijing = utc_time.astimezone(beijing) 进行修改时,时区是不会改动仍是 2019-06-26 11:55:18.150625 只是后面加了 +08:00。

用 time_tokyo = utc_time.astimezone(Tokyo) 进行修改时,会先把时区变成 UTC 时间 :2019-06-26 3:55:18.150625+00:00, 再加上东京的时区,变为:2019-06-26 12:55:18.150625+09:00。

2)、正确的时区转换

先把 utc_time 利用 replace(tzinfo=时区对象) 强制更改 时区属性:

utc_time = datetime.utcnow()    # 获取当前 UTC 时间
print(f'UTC时间为:{utc_time}')
local_time = datetime.now()     # 获取当前本地时间
print(f'本地时间为:{local_time}')

utc = timezone.utc                                  # 获取 UTC 的时区对象
utc_time = datetime.utcnow().replace(tzinfo=utc)    # 强制转换加上 UTC 时区。此处敲黑板,需要特别注意。
                                                    # replace的tzinfo参数为时区对象,所以
                                                    # 也可以这样 replace(tzinfo=timezone(timedelta(hours=0))
print(f'1、强制更改后的UTC时间为:{utc_time}')

time_beijing = utc_time.astimezone(beijing)
time_newyork = utc_time.astimezone(New_York)
time_tokyo = utc_time.astimezone(Tokyo)
print('2、更改时区为北京后的时间:', time_beijing)
print('3、获取时区信息:', time_beijing.tzinfo)
print('4、更改时区为东京后的时间:', time_tokyo)
print('5、获取时区信息:', time_tokyo.tzinfo)
print('6、更改时区为纽约后的时间:', time_newyork)
print('7、获取时区信息:', time_newyork.tzinfo)

运行结果为:

UTC时间为:2019-06-26 12:29:10.178907
本地时间为:2019-06-26 20:29:10.178907
 
1、强制更改后的UTC时间为:2019-06-26 12:29:10.178907+00:00    # 这里为上面的UTC时间加上了 +00:00
2、更改时区为北京后的时间: 2019-06-26 20:29:10.178907+08:00   # 此时的北京时间就和上面的本地时间一样了。
3、获取时区信息: UTC+08:00
4、更改时区为东京后的时间: 2019-06-26 21:29:10.178907+09:00   # 东京时间比北京时间早1小时,也正常。
5、获取时区信息: UTC+09:00
6、更改时区为纽约后的时间: 2019-06-26 08:29:10.178907-04:00   # 纽约时间比UTC时间晚4小时,也是正常的。
7、获取时区信息: UTC-04:00

注:astimezone()修改 时区 会对应的调整日期和时间。

replace(tzinfo=时区) 只是修改时区属性,不会修改日期和时间。

此时如果 用 timestamp() 把日期转换为时间戳,那么它们三个的时间戳应该都一样的,即:

time_beijing.timestamp() == time_tokyo.timestamp() == time_newyork.timestamp()

因为这三个时间只是不同时区的表示方法,对应的都是当时的那一刻时间。

python2和python3的datetime时区问题:timezone时间转换

1.python2.7

python2.7的datetime包没有timezone类,用第三方包解决

pip install pytz
import pytz  
from datetime import datetime
 
u = datetime.utcnow()
u = u.replace(tzinfo=pytz.utc) #NOTE: it works only with a fixed utc offset
 
print(u)   # prints UTC time
>>> '2020-10-23 06:33:19+00:00'
print(u.astimezone(pytz.timezone("America/New_York")))     # prints another timezone
>>> '2020-10-23 02:33:19-04:00'
 
t = datetime.now(tz=pytz.timezone('Asia/Shanghai')
print(t)
>>> '2020-10-23 14:33:19+08:00'
print(t.astimezone(pytz.timezone("America/New_York")))
>>> '2020-10-23 02:33:19-04:00'
 
t = datetime.now()
print(t)
>>> '2020-10-23 14:33:19'
print(t.astimezone(pytz.timezone("America/New_York")))
>>> ValueError: astimezone() cannot be applied to a naive datetime
  • datetime.replace(tzinfo=tz) 将timezone信息赋给datetime,而不改变日期的值,datetime从一个无时区的状态变成有时区的状态
  • datetime.astimezone(tzinfo=tz) 时区转换,datetine转换成新时区的值,执行此方法的datetime必须声明过timezone,否则会报cannot apply to a naive datetime

2. python3

python3.2之后的datetime包引入了timezone的类

from datetime import datetime, timedelta, timezone
 
u = datetime.utcnow()
u = u.replace(tzinfo=timezone.utc) #NOTE: it works only with a fixed utc offset
 
print(u)   # prints UTC time
>>> '2020-10-23 06:33:19+00:00'
print(u.astimezone(timezone(timedelta(hours=-4), "America/New_York")))    # prints another timezone
>>> '2020-10-23 02:33:19-04:00'
datetime.timezone(offset, name=None)¶ 

offset是timedelta对象,取值限制在 -timedelta(hours=24) and timedelta(hours=24)之间

以上为个人经验,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • 浅谈Python3中datetime不同时区转换介绍与踩坑

    最近的项目需要根据用户所属时区制定一些特定策略,学习.应用了若干python3的时区转换相关知识,这里整理一部分记录下来. 下面涉及的几个概念及知识点: GMT时间:Greenwich Mean Time, 格林尼治平均时间 UTC时间:Universal Time Coordinated 世界协调时,可以认为是更精准的GMT时间,但两者误差极小,在1s以内,一般可视为等同 LMT:Local Mean Time, 当地标准时间 Python中的北京时间:Python的标准timezone中信息

  • Python datetime 如何处理时区信息

    在 Python 常用日期处理 -- 内置模块 datetime 探讨了 Python 如何使用 datetime, 如果是一个跨时区的应用(Web 应用都是),就不能只存储一个时间而不带时区,如此,全球用户将会看到一个相同的时间字符串,白天黑夜就错乱了.比说用户信息的更新时间存储为 2020-07-07 13:46:08, 上海的用户和芝加哥的用户看到的是同一个时间字符串,实质上却相差好多个小时. 我们可以这么做,在服务端只存储一个 Timestamp 长整型值或 UTC 时间,Timesta

  • python中的时区问题

    问题背景 使用 Python 进行了许久的开发,一直没有踩到时区的坑,最近新的业务中引入了比较多的服务,而且使用 grpc 进行数据通讯,不幸踩到了时区的坑,果然偷的懒最终还是会有报应的,于是梳理下对应的时区问题,同时发现系统中之前的数据库 Mongo 中的时区问题,一起整理如下. 基础概念 几个时间概念 首先是几个常见的时间概念 GMT 时间:格林威治时间,基准时间 UTC 时间:Coordinated Universal Time,全球协调时间,更精准的基准时间,与 GMT 基本等同 CST

  • python标准库 datetime的astimezone设置时区遇到的坑及解决

    目录 datetime的astimezone设置时区遇到的坑 1.datetime 获取当前的本地日期和UTC日期 2.现在定义时区对象 3.现在在 UTC 的时间基础上,用 astimezone() 修改时区 python2和python3的datetime时区问题:timezone时间转换 1.python2.7 2. python3 datetime的astimezone设置时区遇到的坑 datetime有四个主要的模块: 1.date 处理年.月.日. 2.time 处理时.分.秒和分数

  • Python标准库datetime date模块的详细介绍

    目录 前言 1.定义 1.2.常见错误 2.date类常用的函数 2.1.获取当期日期 2.2.格式化日期 2.2.1.ctime() 2.2.2.datetime.date对象 2.2.3.replace(self, year=None, month=None, day=None) 2.2.4.格式化日期 2.3.ISO标准格式日期 2.3.1.获取符合ISO标准格式的日期字符串的星期几(1~7) 2.3.2.返回日期或者时间对象的星期几(0~6) 2.3.3.根据时间戳计算日期 2.3.4.

  • Python标准库datetime之datetime模块用法分析详解

    目录 1.日期时间对象 2.创建日期时间对象 2.1.通过datetime.datetime.utcnow()创建 2.2.通过datetime.datetime.today()函数创建 2.3.通过datetime.datetime.now()创建 2.4.通过datetime.datetime()创建 2.5.查看创建的对象 2.6.查看datetime可以处理的最大的日期时间对象及最小的日期时间对象 3.日期事件对象的属性 4.日期时间对象转换为时间元组 5.将日期时间对象转化为公元历开始

  • 一篇文章带你了解python标准库--datetime模块

    目录 1. datetime模块介绍 1.1 datetime模块包含的类 1.2 datetime模块中包含的常量 2. datetime实例的方法 3. 日期格式化符号 总结 1. datetime模块介绍 1.1 datetime模块包含的类 1.2 datetime模块中包含的常量 2. datetime实例的方法 案例代码 import locale from datetime import datetime,date,time locale.setlocale(locale.LC_C

  • C站最全Python标准库总结,你想要的都在这里

    python标准库思维导图: 1.操作系统接口 os模块提供了不少与操作系统相关联的函数. >>> import os >>> os.getcwd() # 返回当前的工作目录 'C:\\Python34' >>> os.chdir('/server/accesslogs') # 修改当前的工作目录 >>> os.system('mkdir today') # 执行系统命令 mkdir 0 建议使用 "import os&qu

  • Python标准库之日期、时间和日历模块

    一.time时间模块 import time 1 .时间戳:time.time() 时间戳(timestamp):时间戳表示的是从1970年1月1日00:00:00开始按秒计算的偏移量. 时间戳单位最适于做日期运算.但是1970年之前的日期就无法以此表示了.太遥远的日期也不行,UNIX和Windows只支持到2038年. time_stamp = time.time() print(time_stamp, type(time_stamp)) # 1574923359.1739607 <class

  • Python标准库calendar的使用方法

    目录 Calendar calendar.Calendar(firstweekday=0)类 calendar.TextCalendar(firstweekday=0) calendar.HTMLCalendar(firstweekday=0)  此模块允许你输出类似Unix cal程序的日历,并提供与日历相关的其他有用功能.值得注意的是,默认情况下,这些日历将星期一作为一周的第一天,将星期日作为一周的最后一天(欧洲惯例).不过,我们可以使用setfirstweekday()方法来设置一周的第一

  • Python标准库time使用方式详解

    目录 1.time库 1.1.获取格林威治西部的夏令时地区的偏移秒数 1.2.时间函数 1.3.格式化时间.日期 1.4.单调时钟 1.time库 时间戳(timestamp)的方式:通常来说,时间戳表示的是从1970年1月1日00:00:00开始按秒计算的偏移量 结构化时间(struct_time)方式:struct_time元组共有9个元素 格式化的时间字符串(format_string),时间格式的字符串 1.1.获取格林威治西部的夏令时地区的偏移秒数 如果该地区在格林威治东部会返回负值(

  • Python标准库urllib2的一些使用细节总结

    Python 标准库中有很多实用的工具类,但是在具体使用时,标准库文档上对使用细节描述的并不清楚,比如 urllib2 这个 HTTP 客户端库.这里总结了一些 urllib2 的使用细节. 1.Proxy 的设置 2.Timeout 设置 3.在 HTTP Request 中加入特定的 Header 4.Redirect 5.Cookie 6.使用 HTTP 的 PUT 和 DELETE 方法 7.得到 HTTP 的返回码 8.Debug Log Proxy 的设置 urllib2 默认会使用

  • Python标准库之collections包的使用教程

    前言 Python为我们提供了4种基本的数据结构:list, tuple, dict, set,但是在处理数据量较大的情形的时候,这4种数据结构就明显过于单一了,比如list作为数组在某些情形插入的效率会比较低,有时候我们也需要维护一个有序的dict.所以这个时候我们就要用到Python标准库为我们提供的collections包了,它提供了多个有用的集合类,熟练掌握这些集合类,不仅可以让我们让写出的代码更加Pythonic,也可以提高我们程序的运行效率. defaultdict defaultd

随机推荐