美文网首页
django utc 时区问题

django utc 时区问题

作者: 技术创造未来 | 来源:发表于2018-08-27 16:22 被阅读0次

    Django时间问题

    在用django1.8版本做项目的时候遇到时间的存储与读取不一致的问题,网上找了很多帖子,但都没有讲明白。本文将在项目中遇到的问题及如何解决的尽可能详细的记录下来,当然本文参考了网上大量相关文章。

    在django1.4以后,存在两个概念:naive time 与 active time。简单点讲,naive time就是不带时区的时间,相关Active time就是带时区的时间。举例来说,使用datetime.datetime.utcnow()、datetime.datetime.now()输出的类似2015-05-11 09:10:33.080451就是不带时区的时间(naive time),而使用django.util.timezone.now()输出的类似2015-05-11 09:05:19.936835+00:00的时间就是带时区的时间(Active time),其中+00:00表示的就是时区相对性。

    另外一个概念UTC时间。这里不做过多介绍,需要知晓的是UTC时间表示的是格林尼治平均时即可,即零区时间。而北京时间表示的是东八区时间,即UTC+8。

    下面列出了几个常见的时区问题,并提供相关原因,如有不对,欢迎指出。

    问题一:三个时间datetime.datetime.now()、datetime.datetime.utcnow()与django.util.timezone.now()的区别

    datetime.datetime.now():输出的永远是本地时间(naive time)与配置无任任何关系。datetime.datetime.utcnow():如果setting中配置USE_TZ=True则输出的是UTC时间(naive time),如果setting中配置USE_TZ=False,则该输出时间与datetime.datetime.now()完全相同。django.util.timezone.now():如果setting中配置USE_TZ=True则输出的是UTC时间(active time),如果配置USE_TZ=False,则与datetime.datetime.now()完全相同。

    问题二:django存储到数据库的时间比本地时间小8个小时?

    首先要明确的一点,Django1.4版本之前,对时区毫无概概念,对时间的存取、展示不做任何处理,数据库里存储的通常是本地时间,当然都是naive time。

    Django在1.4版本之后存储如果设置了USE_TZ=True,则存储到数据库中的时间永远是UTC时间。这时如果settings里面设置了USE_TZ=True与TIME_ZONE = 'UTC',用datetime.datetime.now()获取的时间django会把这个时间当成UTC时间存储到数据库中去。如果修改设置为USE_TZ=True与TIME_ZONE = 'Asia/Shanghai',用datetime.datetime.now()获取的时间由于不带时区,django会把这个时间当成Asia/Shanghai时间,即东八区时间,然后django会把这个时间转成带时区UTC时间存储到数据库中去,而读的时候直接按UTC时间读出来,这就是网上很多人遇到的存储到数据库中的时间比本地时间会小8个小时的原因。

    问题三:DateTimeField role_cost_history.cost_time received a naive datetime (2015-05-12 19:59:01.259517) while time zone support is active?

    这个问题是因为如果设置了USE_TZ=True之后,model里面认为DateTimeField使用UTC时间(带时区的时间),这时用datetime.datetime.now()获取的时间是不带时区的就会报这个问题。

    问题四:django.util.timezone.now()输出时间比本地时间小8个小时

    只要设置了USE_TZ=True,django.util.timezone.now()输出地永远是UTC时间,不管你设置的TIME_ZONE是什么。如果USE_TZ=False,则django.util.timezone.now()输出等同于datetime.datetime.now(),也不管TIME_ZONE设置的是什么。

    问题五:模板显示时间

    在设置了USE_TZ=True之后,如果设置了TIME_ZONE = 'Asia/Shanghai',尽管数据库中存储的是UTC时间,但在模板显示的时候,会转成TIME_ZONE所示的本地时间进行显示。

    建议:为了统一时间,在django开发时,尽量使用UTC时间,即设置USE_TZ=True,TIME_ZONE = 'Asia/Shanghai',并且在获取时间的时候使用django.util.timezone.now()。因为后台程序使用时间时UTC时间就能满足,也能保证证模板时间的正确显示。

    python datetime timezone 时区转化

    from datetime import datetime, timedelta, timezone

    utc_dt = datetime.utcnow().replace(tzinfo=timezone.utc)

    print(utc_dt)

    cn_dt = utc_dt.astimezone(timezone(timedelta(hours=8)))

    print(cn_dt)

    jan_dt = utc_dt.astimezone(timezone(timedelta(hours=9)))

    print(jan_dt)

    cn_2_jan_dt = cn_dt.astimezone(timezone(timedelta(hours=9)))

    print(cn_2_jan_dt)

    import datetime

    nowTime=datetime.datetime.now().strftime('%Y-%m-%d %H:%M:%S')#现在

    pastTime = (datetime.datetime.now()-datetime.timedelta(hours=1)).strftime('%Y-%m-%d %H:%M:%S')#过去一小时时间

    afterTomorrowTime = (datetime.datetime.now()+datetime.timedelta(days=2)).strftime('%Y-%m-%d %H:%M:%S')#后天

    tomorrowTime = (datetime.datetime.now()+datetime.timedelta(days=1)).strftime('%Y-%m-%d %H:%M:%S')#明天

    print('\n',nowTime,'\n',pastTime,'\n',afterTomorrowTime,'\n',tomorrowTime)


    Django 时间与时区设置问题

    在Django的配置文件settings.py中,有两个配置参数是跟时间与时区有关的,分别是TIME_ZONE和USE_TZ

    如果USE_TZ设置为True时,Django会使用系统默认设置的时区,即America/Chicago,此时的TIME_ZONE不管有没有设置都不起作用。

    如果USE_TZ 设置为False,而TIME_ZONE设置为None,则Django还是会使用默认的America/Chicago时间。若TIME_ZONE设置为其它时区的话,则还要分情况,如果是Windows系统,则TIME_ZONE设置是没用的,Django会使用本机的时间。如果为其他系统,则使用该时区的时间,入设置USE_TZ = False, TIME_ZONE = 'Asia/Shanghai', 则使用上海的UTC时间。

    一、查看和修改Linux的时区

    1. 查看当前时区

    命令 : "date -R"

    2. 修改设置Linux服务器时区

    方法 A

    命令 : "tzselect"

    方法 B 仅限于RedHat Linux 和 CentOS

    命令 : "timeconfig"

    方法 C 适用于Debian

    命令 : "dpkg-reconfigure tzdata"

    3. 复制相应的时区文件,替换系统时区文件;或者创建链接文件

    cp /usr/share/zoneinfo/$主时区/$次时区 /etc/localtime

    例如:在设置中国时区使用亚洲/上海(+8)

    cp /usr/share/zoneinfo/Asia/Shanghai /etc/localtime

    4.注意

    GMT(Greenwich Mean Time,格林威治标准时间):

    是指位于英国伦敦郊区的皇家格林尼治天文台的标准时间,因为本初子午线被定义在通过那里的经线。

    UTC(Universal Time/Temps

    Cordonné 世界标准时间)

    CST(Central Standard Time 國家標準時間,一說中原標準時間); 中国标准时间(China

    Standard Time)

    GMT + 8 = UTC + 8 = CST

    二、查看和修改Linux的时间

    1、date

    查看系统时间

    # date

    设置系统时间

    # date --set “07/07/06 10:19" (月/日/年 时:分:秒)

    将当前时间和日期写入BIOS,避免重启后失效

    命令 : "hwclock -w"

    2、hwclock/clock

    查看硬件时间

    # hwclock --show

    或者 # clock --show

    设置硬件时间

    # hwclock --set --date="07/07/06 10:19" (月/日/年 时:分:秒)

    或者 # clock --set --date="07/07/06 10:19" (月/日/年 时:分:秒)

    3、硬件时间和系统时间的同步

    按照前面的说法,重新启动系统,硬件时间会读取系统时间,实现同步,但是在不重新启动的时候,需要用hwclock或clock命令实现同步。

    硬件时钟与系统时钟同步:

    # hwclock --hctosys(hc代表硬件时间,sys代表系统时间)

    或者

    # clock --hctosys

    系统时钟和硬件时钟同步: # hwclock --systohc 或者 # clock --systohc

    针对中国时区,修改操作如下

    1.   修改文件/etc/sysconfig/clock内容:

    ZONE=Asia/Shanghai UTC=false ARC=false

    2.     rm /etc/localtime

    ln -sf /usr/share/zoneinfo/Asia/Shanghai /etc/localtime

    查看系统时区:

    timedatectl | grep "Time zone"

    相关文章

      网友评论

          本文标题:django utc 时区问题

          本文链接:https://www.haomeiwen.com/subject/ztltwftx.html