Golang MySQL 时区的坑
UULU · · 8874 次点击 · · 开始浏览在使用go-sql-driver操作 mysql 时,遇到了由于 mysql 与程序不在同一时区引发的问题
import "database/sql"
import _ "github.com/go-sql-driver/mysql"
db, err := sql.Open("mysql", "user:password@/dbname")
问题
想要把 time.Time 直接存储入库,需要开启解析时间parseTime
db, err := sql.Open("mysql", "user:password@/dbname?charset=utf8mb4&parseTime=true")
golang 程序里 time.Time 为
2018年12月24日 18:00:00 CST转为 UTC 存储到 mysql
2018年12月24日 10:00:00golang 从 mysql 获取解析成 time.Time 为
2018年12月24日 10:00:00 UTC
以上问题可以通过设置loc=Local解决
db, err := sql.Open("mysql", "user:password@/dbname?charset=utf8mb4&parseTime=true&loc=Local")
golang 程序里 time.Time 为
2018年12月24日 18:00:00 CST转为 UTC 存储到 mysql
2018年12月24日 18:00:00golang 从 mysql 获取解析成 time.Time 为
2018年12月24日 18:00:00 CST
但要注意,这个方法不会修改连接的time_zone属性,而是在go-sql-driver程序里对 time.Time 做了时区转换,所以会遗留一个问题,如果在我们的程序里执行 SQL 语句,UNIX_TIMESTAMP(NOW()) - UNIX_TIMESTAMP(created_at),会出现意想不到的问题,因为NOW()取到的仍然是 UTC 时间,而created_at为 GST 时间
这个问题可以通过同时指定loc=true&time_zone=*来解决
需要用url.QueryEscape对timezone进行编码,单引号不可省略
但这种方法我们需要手动配置时区名,这样并不方便,所以建议避免直接调用 MYSQL 的时间函数
timezone := "'Asia/Shanghai'"
db, err := sql.Open("mysql", "user:password@/dbname?charset=utf8mb4&parseTime=true&loc=Local&time_zone=" + url.QueryEscape(timezone))
参考文档
有疑问加站长微信联系(非本文作者)
入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889
关注微信- 请尽量让自己的回复能够对别人有帮助
- 支持 Markdown 格式, **粗体**、~~删除线~~、
`单行代码` - 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
- 图片支持拖拽、截图粘贴等方式上传
收入到我管理的专栏 新建专栏
在使用go-sql-driver操作 mysql 时,遇到了由于 mysql 与程序不在同一时区引发的问题
import "database/sql"
import _ "github.com/go-sql-driver/mysql"
db, err := sql.Open("mysql", "user:password@/dbname")
问题
想要把 time.Time 直接存储入库,需要开启解析时间parseTime
db, err := sql.Open("mysql", "user:password@/dbname?charset=utf8mb4&parseTime=true")
golang 程序里 time.Time 为
2018年12月24日 18:00:00 CST转为 UTC 存储到 mysql
2018年12月24日 10:00:00golang 从 mysql 获取解析成 time.Time 为
2018年12月24日 10:00:00 UTC
以上问题可以通过设置loc=Local解决
db, err := sql.Open("mysql", "user:password@/dbname?charset=utf8mb4&parseTime=true&loc=Local")
golang 程序里 time.Time 为
2018年12月24日 18:00:00 CST转为 UTC 存储到 mysql
2018年12月24日 18:00:00golang 从 mysql 获取解析成 time.Time 为
2018年12月24日 18:00:00 CST
但要注意,这个方法不会修改连接的time_zone属性,而是在go-sql-driver程序里对 time.Time 做了时区转换,所以会遗留一个问题,如果在我们的程序里执行 SQL 语句,UNIX_TIMESTAMP(NOW()) - UNIX_TIMESTAMP(created_at),会出现意想不到的问题,因为NOW()取到的仍然是 UTC 时间,而created_at为 GST 时间
这个问题可以通过同时指定loc=true&time_zone=*来解决
需要用url.QueryEscape对timezone进行编码,单引号不可省略
但这种方法我们需要手动配置时区名,这样并不方便,所以建议避免直接调用 MYSQL 的时间函数
timezone := "'Asia/Shanghai'"
db, err := sql.Open("mysql", "user:password@/dbname?charset=utf8mb4&parseTime=true&loc=Local&time_zone=" + url.QueryEscape(timezone))