zoukankan      html  css  js  c++  java
  • Django models通过DateTimeField保存到MySQL的时间的时区问题

    最近开始使用Django开发一些系统,在models.py中设置一些数据库表结构并给日期时间字赋初值,不过在使用的过程中,遇到一点问题。问题是,我本来服务器使用的市区是“Asia/Shanghai”(+08:00),然后用datetime.now保存到数据库中时,里面的时间总是比我的系统时间相差8小时(感觉是UTC时间),但是我也查看了MySQL数据库中的时区是“+08:00”的(在MySQL中“SET GLOBAL time_zone = ‘+08:00′;”语句可以设置时区)。这到底是什么原因呢?今天还花了不少时间来弄清楚这个问题。

    我在models.py中定义的一个类如下(对于数据库中一张表):

    class TestSuite(models.Model):  id = models.AutoField(primary_key=True)  name = models.CharField(max_length=40, blank=True)  description = models.CharField(max_length=255, blank=True)  bu = models.CharField(max_length=40, blank=True)  is_enabled = models.BooleanField(default=True)  create_time = models.DateTimeField(default=datetime.now)  update_time = models.DateTimeField(default=datetime.now)


    Django还有一些warning打印出来:/Users/jay/workspace/te/env/lib/
    python2.7/site-packages/django/db/models/fields/__init__.py:903: RuntimeWarning: DateTimeField TestSuite.update_time received a naive datetime (2014-06-15 14:38:37.873873) while time zone support is active. RuntimeWarning) 这里的create_time和update_time存到数据库中的时间总是UTC时区的时间(比北京时间慢了8个小时)。

    这个warning的原因是,Django配置为使用timezone的datetime格式,而datetime.now是不含timezone信息的。

    如果不需要在程序中特别处理时区(timezone-aware),在Django项目的settings.py文件中,可以直接设置为“USE_TZ = False”就省心了。然后,在models.py中简单的设置为“ create_time = models.DateTimeField(auto_now_add=True)”和“update_time = models.DateTimeField(auto_now=True)”。

    如果还要保持USE_TZ=True,则可设置为“default=datetime.now().replace(tzinfo=utc)” 。

    我还试图去通过,django.utils.timezone.now()函数来获取当前时间,我还以为settings.py中已经设置“TIME_ZONE = ‘Asia/Shanghai’”,就可以得到正确的时区和时间;可惜它依然得到的是UTC的时间,翻看它的实现就一下明白了,如下:

    def now():  """  Returns an aware or naive datetime.datetime, depending on settings.USE_TZ.  """ if settings.USE_TZ:   # timeit shows that datetime.now(tz=utc) is 24% slower return datetime.utcnow().replace(tzinfo=utc) else:   return datetime.now()


    还是试图通过timezone来使用系统本地时间和时区,比如:default=timezone.localtime(timezone.now()) ,结果发现,也是不行的,存到My
    SQL中也是UTC时区的日期时间。得看下Django相应的实现代码了,DateTimeField与MySQL数据库交互的时候,是否都先换为UTC的时间再更新到数据库的。另外,如果USE_TZ=True,那么设置auto_now_add=True(或auto_now=True)可以达到我的目的吗?答案也是否定的,翻看了一下Django对auto_now_add=True的处理就明白了,它就是对其赋值为 django.utils.timezone.now() 。

    参考资料:

    https://docs.djangoproject.com/en/1.4/ref/settings/#time-zone

    https://docs.djangoproject.com/en/dev/topics/i18n/timezones/#usage

  • 相关阅读:
    人生转折点:弃文从理
    人生第一站:大三暑假实习僧
    监听器启动顺序和java常见注解
    java常识和好玩的注释
    182. Duplicate Emails (Easy)
    181. Employees Earning More Than Their Managers (Easy)
    180. Consecutive Numbers (Medium)
    178. Rank Scores (Medium)
    177. Nth Highest Salary (Medium)
    176. Second Highest Salary(Easy)
  • 原文地址:https://www.cnblogs.com/zhaojia-dream/p/4120603.html
Copyright © 2011-2022 走看看