• pytz库时区的坑(转)


    add by zhj: 推荐使用标准库中的dateutil代替pytz

    原文:https://www.hongweipeng.com/index.php/archives/1803/

    起步

    在django框架中,用的是 pytz 库处理时区问题,所以我也尝试用这个库来处理。但发现了一个奇怪的问题:

    1
    2
    3
    4
    5
    import datetime
    import pytz
     
    dt = datetime.datetime(2019,6,20, 12, tzinfo=pytz.timezone('Asia/Shanghai'))
    print(dt) # 2019-06-20 12:00:00+08:06

    为什么多出了 6 分钟来?

    原因

    这是因为 pytz 里保存是本地时间。

    1
    2
    3
    fmt = '%Y-%m-%d %H:%M:%S %Z%z'
    dt = datetime.datetime(2019,6,20, 12, tzinfo=pytz.timezone('Asia/Shanghai'))
    print(dt.strftime(fmt)) # 2019-06-20 12:00:00 LMT+0806

    LMT 即 Local Mean Time 本地时间,也就是说 'Asia/Shanghai' 这个地区比utc多了8小时零6分钟,并不是北京时间。

    解决

    所以 pytz 提供了 localize() 方法来纠正这个问题,但传入的得是不带时区的日期对象:

    1
    2
    3
    4
    5
    6
    7
    8
    cn_zone = pytz.timezone('Asia/Shanghai')
    dt = cn_zone.localize(dt = datetime.datetime(2019,6,20, 12))
    print(dt) # 2019-06-20 12:00:00+08:00
    print(dt.strftime(fmt)) # 2019-06-20 12:00:00 CST+0800
     
    # 或者
    dt = datetime.datetime(2019,6,20, 12)
    print(dt.astimezone(cn_zone)) # 2019-06-20 12:00:00 CST+0800

    这里推荐的还是用 astimezone 的方式,django 也是采用这种处理方式。

    时区转换

    以从北京时间转纽约时间为例,已知结果它们之间应该相差12小时。

    1
    2
    3
    4
    dt = datetime.datetime(2019,6,20, 12)
     
    print(dt.astimezone(tz=cn_zone)) # 2019-06-20 12:00:00+08:00
    print(dt.astimezone(tz=cn_zone).astimezone(ny_zone)) # 2019-06-20 12:00:00-04:56

    额外

    另一个解决时区问题就是使用标准库的 dateutil 工具。官方内置,值得信赖。它支持在创建日期对象的时候进行设置,更为方便:

    1
    2
    3
    4
    5
    6
    7
    8
    cn = tz.gettz('Asia/Shanghai')
     
    aware_dt = datetime.datetime(2019,6,20, 12, tzinfo=cn)
    print(aware_dt ) # 2019-06-20 12:00:00+08:00
     
    # 时区转换(从北京时间转到纽约时间)
    ny = tz.gettz('America/New_York')
    print(aware_dt.astimezone(tz=ny)) # 2019-06-20 00:00:00-04:00

    我更喜欢这种方式。

  • 相关阅读:
    扫雷游戏

    set
    map
    认识了个外国友人!
    插入排序算法
    复习 C语言
    有关于我的一点想法
    linux下的内存分布
    C语言-将输入的字符输出并将多个空格按照一个空格的方式输出
  • 原文地址:https://www.cnblogs.com/ajianbeyourself/p/11982529.html
Copyright © 2020-2023  润新知