Golang MySQL 时区的坑 - Go语言中文社区

Golang MySQL 时区的坑


在使用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:00

  • golang 从 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:00

  • golang 从 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 时间

方法 1(不推荐)

可以通过同时指定loc=true&time_zone=*来解决,需要用url.QueryEscapetimezone进行编码,单引号不可省略

timezone := "'Asia/Shanghai'"
db, err := sql.Open("mysql", "user:password@/dbname?charset=utf8mb4&parseTime=true&loc=Local&time_zone=" + url.QueryEscape(timezone))

方法 2(推荐)

上面的方法我们需要手动配置时区名,这样并不方便,而且不同机器获取到的时区名会有差异,所以我建议避免直接调用 MYSQL 的时间函数,而是用程序里的时间拼接 SQL 语句

// 修改 "UNIX_TIMESTAMP(NOW()) - UNIX_TIMESTAMP(created_at)" 为
fmt.Sprintf(`UNIX_TIMESTAMP("%s") - UNIX_TIMESTAMP(created_at)`, time.Now().Format("2006-01-02 15:04:05"))

参考文档

https://github.com/go-sql-driver/mysql

版权声明:本文来源简书,感谢博主原创文章,遵循 CC 4.0 by-sa 版权协议,转载请附上原文出处链接和本声明。
原文链接:https://www.jianshu.com/p/030b880ecc5e
站方申明:本站部分内容来自社区用户分享,若涉及侵权,请联系站方删除。
  • 发表于 2020-01-09 22:18:52
  • 阅读 ( 2044 )
  • 分类:数据库

0 条评论

请先 登录 后评论

官方社群

GO教程

猜你喜欢