win11下的docker中有个mysql。今天发现插入数据的时间不正确。后来发现原来是docker容器中的时间不正确。于是尝试了各种修改,什么run -e TZ="${tzutil /g}",TZ="Asia/Shanghai",还有初始化时带--mysqld一类的,都无效。
后来偶然间发现wsl中的docker-desktop中的时间和mysql中的一致,才突然想到,其实容器是运行在docker-desktop这个发行版中的,所以它的环境就是各容器的环境,于是进入docker-desktop,执行:
#在wsl中修改系统时间
date -s "2025-06-03 13:50"
使用date修改为当前时间即可。
/************************************ 又发现点问题,更新一下 *****************************************/
根据上面的修改,改完后,过一会时间又恢复了不正确的时间。初步原因如下:
docker-desktop是基于alpine linux。它直接date -s修改时间后,过一会会自动同步为UTC标准时间。如果查看win平台,会发现win其实是使用了UTC+8,它是在utc的基础上偏移8小时,获得当前区域时间。而docker-desktop只是同步了UTC,而未偏移,这一点使用
# 查看标准硬件时间
hwclock -r
可以发现,它后面的延迟是0,即标准UTC时间,而不是UTC+8。
于是需要将alpine中的时区替换为CST8,这一步中,如果你的网络足够好,可以撑杆跳,那直接安装tzdata即可。如果像我这一样,根本连不上,而wsl中恰好还有一个ubuntu的时候,方案就有了。步骤为:从ubuntu中将/usr/share/zoneinfo/Asia文件夹中的内容,复制到docker-desktop中,再创建一个软链接即可。
# 登陆wsl,注意这里使用root用户,否则就会像我一样,被要求输入一个谁也不知道的密码
wsl -u root
# 登陆后,复制文件
cd /usr/share/zoneinfo/
tar -czvf Asia.tar.gz Asia/
mv Asia.tar.gz /mnt/c/tmp/
exit
# 这样就将Asia.tar.gz复制到c:\tmp文件中了
# 登陆docker对应的linux
wsl -d docker-desktop
mv /mnt/host/c/tmp/Asia.tar.gz /usr/share/zoneinfo #如果没有这个目录,建一个
cd /usr/share/zoneinfo
tar -xzvf Asia.tar.gz
#复制完成了
# 创建一个链接
ln -sf /usr/share/zoneinfo/Asia/Shanghai /etc/localtime
如果执行完了,都没有报错,则执行date后,会发现时区变为了CST,时间也正确了。
但是!!!容器中的时间和docker-desktop发行版中的时区是独立的。于是创建新的容器
docker run -d --name mysql \
-v volume_name:/var/lib/mysql \ # 自己的volume的设置
-e MYSQL_ROOT_PASSWORD=root \
-e TZ=Asia/Shanghai \ # 这个是关键
-p 3306:3306 mysql:8 \
--lower-case-table-names=1
然后再select now()查看一下,发现时间一致了。