Django设置TIME_ZONE为中国,及其规则与善后问题

Django默认的timezone是TIME_ZONE = ‘America/Chicago’

现在要改成我们中国的时区 只需编辑settings.py文件

把time_zone的值改成TIME_ZONE=’Asia/Shanghai’即可

Django 启用 USE_TZ = True 的善后工作

Django 1.4 之后,时区的问题总算解决了。虽然 pytz 库 是可选的,但 pytz 可以帮助 Django 识别 TIME_ZONE = 'Asia/Shanghai' 对应的时区是 UTC+8 时间,因此建议启用 USE_TZ = True 的同时也安装 pytz。

1.4 之前,Django 对时区毫无概念,对时间的存取、展示不做任何处理,数据库里存储的通常是本地时间(local time)。1.4 之后,在 settings 里面设置 USE_TZ = True 即让 Django 内部把时间全部当成 UTC 时间(北京时间为 UTC+8 )对待。

但这么做还没完,还有一些琐碎的善后工作要做 — 数据和老代码的迁移,不然时间的存取、展示都会有差错。

迁移数据库

如果之前设置了 TIME_ZONE = 'Asia/Shanghai',那么实际存储在数据库中的是 UTC+8 的时间。启用USE_TZ = True 我们要把数据库中所有的时间戳(timestamp)字段全部转换成 UTC 时间。

转换到 UTC 时间有两个方法(注:PostgreSQL 有专门的方法):

  1. 使用 manage.py shell 进入 Django shell 环境,用 Model.objects.all() 循环遍历包含DateTimeField 字段的模型,减去 timedelta(hours=8)
  2. 使用数据库 UPDATE 语句;

第一个方法不推荐使用,因为 ORM 的 save() 方法会触发 post_save signal,可能会产生不希望看到的副作用。

第二个方法是安全的(注意备份数据库),MySQL 下可以用如下 SQL 语句更新时间戳:

UPDATE `your_model` SET `last_modified` = DATE_SUB(`last_modified`, INTERVAL 8 HOUR) WHERE 1;

PostgreSQL 的时间字段原生支持时区,所以处理方法不太一样,详见 Django 官方文档 Time zones

迁移代码

启用 USE_TZ = True 后,处理时间方面,有两条 “黄金法则”:

  1. 保证存储到数据库中的是 UTC 时间;
  2. 在函数之间传递时间参数时,确保时间已经转换成 UTC 时间;

比如,通常获取当前时间用的是:

import datetime

now = datetime.datetime.now()

启用 USE_TZ = True 后,需要写成:

import datetime
from django.utils.timezone import utc

now = datetime.datetime.utcnow().replace(tzinfo=utc)

保证 now 变量存放的是 UTC 时间。

再如 fromtimestamp() 这个函数,启用 USE_TZ = True 后应使用 utcfromtimestamp() 函数替代。

模板

除非应用支持用户设置自己所在的时区,通常我们不需要关心模板的时区问题。模板在展示时间的时候,会使用 settings.TIME_ZONE 中的设置自动把 UTC 时间转成 settings.TIME_ZONE 所在时区的时间渲染。

如果确实需要支持用户设置时区,参考 Django 官方文档 Time zones

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值