前言:
在使用 MySQL 的過程中,你可能會遇到時區(qū)相關問題,比如說時間顯示錯誤、時區(qū)不是東八區(qū)、程序取得的時間和數據庫存儲的時間不一致等等問題。其實,這些問題都與數據庫時區(qū)設置有關,本篇文章將從數據庫參數入手,逐步介紹時區(qū)相關內容。
1.log_timestamps 參數介紹
首先說明下log_timestamps
參數并不影響時區(qū),只是設置不同會影響某些日志記錄的時間。該參數主要是控制 error log、slow log、genera log 日志文件中的顯示時間,但不會影響 general log 和 slow log 寫到表 (mysql.general_log, mysql.slow_log) 中的顯示時間。
log_timestamps 是全局參數,可動態(tài)修改,默認使用 UTC 時區(qū),這樣會使得日志中記錄的時間比北京時間慢 8 個小時,導致查看日志不方便??梢孕薷臑?SYSTEM 變成使用系統(tǒng)時區(qū)。下面簡單測試下該參數的作用及修改方法:
# 查看參數值 mysql> show global variables like 'log_timestamps'; +----------------+-------+ | Variable_name | Value | +----------------+-------+ | log_timestamps | UTC | +----------------+-------+ 1 row in set (0.00 sec) # 產生慢日志 mysql> select sleep(10),now(); +-----------+---------------------+ | sleep(10) | now() | +-----------+---------------------+ | 0 | 2020-06-24 17:12:40 | +-----------+---------------------+ 1 row in set (10.00 sec) # 慢日志文件記錄內容 發(fā)現(xiàn)時間是UTC時間 # Time: 2020-06-24T09:12:50.555348Z # User@Host: root[root] @ localhost [] Id: 10 # Query_time: 10.000354 Lock_time: 0.000000 Rows_sent: 1 Rows_examined: 1 SET timestamp=1592989960; select sleep(10),now(); # 修改參數值 再次測試 mysql> set global log_timestamps = SYSTEM; Query OK, 0 rows affected (0.00 sec) mysql> select sleep(10),now(); +-----------+---------------------+ | sleep(10) | now() | +-----------+---------------------+ | 0 | 2020-06-24 17:13:44 | +-----------+---------------------+ 1 row in set (10.00 sec) # 慢日志文件記錄內容 時間是對的 # Time: 2020-06-24T17:13:54.514413+08:00 # User@Host: root[root] @ localhost [] Id: 10 # Query_time: 10.000214 Lock_time: 0.000000 Rows_sent: 1 Rows_examined: 1 SET timestamp=1592990024; select sleep(10),now();
2.time_zone 參數介紹
time_zone
參數用來設置每個連接會話的時區(qū),該參數分為全局和會話級別,可以動態(tài)修改。默認值為 SYSTEM,此時使用的是全局參數 system_time_zone 的值,而 system_time_zone 默認繼承自當前系統(tǒng)的時區(qū),即默認情況下 MySQL 時區(qū)和系統(tǒng)時區(qū)相同。
時區(qū)設置主要影響時區(qū)敏感的時間值的顯示和存儲。包括一些函數(如 now()、curtime())顯示的值,以及存儲在 TIMESTAMP 類型中的值,但不影響 DATE、TIME 和 DATETIME 列中的值,因為這些數據類型在存取時未進行時區(qū)轉換,而 TIMESTAMP 類型存入數據庫的實際是 UTC 的時間,查詢顯示時會根據具體的時區(qū)來顯示不同的時間。
下面我們來測試下 time_zone 參數修改產生的影響:
# 查看linux系統(tǒng)時間及時區(qū) [root@centos ~]# date Sun Jun 28 14:29:10 CST 2020 # 查看MySQL當前時區(qū)、時間 mysql> show global variables like '%time_zone%'; +------------------+--------+ | Variable_name | Value | +------------------+--------+ | system_time_zone | CST | | time_zone | SYSTEM | +------------------+--------+ 2 rows in set (0.00 sec) mysql> select now(); +---------------------+ | now() | +---------------------+ | 2020-06-28 14:31:12 | +---------------------+ 1 row in set (0.00 sec) # 創(chuàng)建測試表、插入部分數據 mysql> CREATE TABLE `time_zone_test` ( -> `id` int unsigned NOT NULL AUTO_INCREMENT COMMENT '自增主鍵', -> `dt_col` datetime DEFAULT NULL COMMENT 'datetime時間', -> `ts_col` timestamp DEFAULT NULL COMMENT 'timestamp時間', -> PRIMARY KEY (`id`) -> ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='time_zone測試表'; Query OK, 0 rows affected, 1 warning (0.07 sec) mysql> insert into time_zone_test (dt_col,ts_col) values ('2020-06-01 17:30:00','2020-06-01 17:30:00'),(now(),now()); Query OK, 2 rows affected (0.01 sec) Records: 2 Duplicates: 0 Warnings: 0 mysql> select * from time_zone_test; +----+---------------------+---------------------+ | id | dt_col | ts_col | +----+---------------------+---------------------+ | 1 | 2020-06-01 17:30:00 | 2020-06-01 17:30:00 | | 2 | 2020-06-28 14:34:55 | 2020-06-28 14:34:55 | +----+---------------------+---------------------+ # 改為UTC時區(qū) 并重新連接 發(fā)現(xiàn)timestamp存儲的時間會隨時區(qū)變化 mysql> set global time_zone='+0:00'; Query OK, 0 rows affected (0.00 sec) mysql> set time_zone='+0:00'; Query OK, 0 rows affected (0.00 sec) mysql> show global variables like '%time_zone%'; +------------------+--------+ | Variable_name | Value | +------------------+--------+ | system_time_zone | CST | | time_zone | +00:00 | +------------------+--------+ 2 rows in set (0.00 sec) mysql> select now(); +---------------------+ | now() | +---------------------+ | 2020-06-28 06:36:16 | +---------------------+ 1 row in set (0.00 sec) mysql> select * from time_zone_test; +----+---------------------+---------------------+ | id | dt_col | ts_col | +----+---------------------+---------------------+ | 1 | 2020-06-01 17:30:00 | 2020-06-01 09:30:00 | | 2 | 2020-06-28 14:34:55 | 2020-06-28 06:34:55 | +----+---------------------+---------------------+ 2 rows in set (0.00 sec) # 改回東八時區(qū),恢復正常 mysql> set global time_zone='+8:00'; Query OK, 0 rows affected (0.00 sec) mysql> set time_zone='+8:00'; Query OK, 0 rows affected (0.00 sec) mysql> show global variables like '%time_zone%'; +------------------+--------+ | Variable_name | Value | +------------------+--------+ | system_time_zone | CST | | time_zone | +08:00 | +------------------+--------+ 2 rows in set (0.00 sec) mysql> select now(); +---------------------+ | now() | +---------------------+ | 2020-06-28 14:39:14 | +---------------------+ 1 row in set (0.00 sec) mysql> select * from time_zone_test; +----+---------------------+---------------------+ | id | dt_col | ts_col | +----+---------------------+---------------------+ | 1 | 2020-06-01 17:30:00 | 2020-06-01 17:30:00 | | 2 | 2020-06-28 14:34:55 | 2020-06-28 14:34:55 | +----+---------------------+---------------------+ 2 rows in set (0.00 sec)
如果需要永久生效,還需寫入配置文件中。例如將時區(qū)改為東八區(qū),則需要在配置文件[mysqld]部分增加一行:default_time_zone = '+8:00'。
3.時區(qū)常見問題及如何避免
時區(qū)設置不妥可能會產生各種問題,下面我們列舉下幾個常見的問題及解決方法:
3.1 MySQL 內部時間不是北京時間
遇到這類問題,首先檢查下系統(tǒng)時間及時區(qū)是否正確,然后看下 MySQL 的 time_zone,建議將 time_zone 改為'+8:00'。
3.2 Java 程序存取的時間與數據庫中的時間相差 8 小時
出現(xiàn)此問題的原因大概率是程序時區(qū)與數據庫時區(qū)不一致導致的。我們可以檢查下兩邊的時區(qū),如果想統(tǒng)一采用北京時間,則可以在 jdbc 連接串中增加 serverTimezone=Asia/Shanghai,并且 MySQL 方面也可以將 time_zone 改為'+8:00'。
3.3 程序時間與數據庫時間相差 13 小時或 14 小時
如果說相差 8 小時不夠讓人驚訝,那相差 13 小時可能會讓很多人摸不著頭腦。出現(xiàn)這個問題的原因是 JDBC 與 MySQL 對 “CST” 時區(qū)協(xié)商不一致。因為 CST 時區(qū)是一個很混亂的時區(qū),有四種含義:
MySQL 中,如果 time_zone 為默認的 SYSTEM 值,則時區(qū)會繼承為系統(tǒng)時區(qū) CST,MySQL 內部將其認為是 UTC+08:00。而 jdbc 會將 CST 認為是美國中部時間,這就導致會相差 13 小時,如果處在冬令時還會相差 14 個小時。
解決此問題的方法也很簡單,我們可以明確指定 MySQL 數據庫的時區(qū),不使用引發(fā)誤解的 CST,可以將 time_zone 改為'+8:00',同時 jdbc 連接串中也可以增加 serverTimezone=Asia/Shanghai。
3.4 如何避免出現(xiàn)時區(qū)問題
如何避免上述時區(qū)問題,可能你心里也有了些方法,簡要總結幾點如下:
可能有的同學說了,我們數據庫中 time_zone 參數選擇的是默認的 SYSTEM 值,也沒有發(fā)生程序時間和數據庫時間不一致的問題。此時是否需要將 time_zone 改為'+8:00'?在這種情況下還是建議將 time_zone 改為'+8:00',特別是經常查詢 TIMESTAMP 字段,因為當 time_zone=system 的時候,查詢 timestamp 字段會調用系統(tǒng)的時區(qū)做時區(qū)轉換,有全局鎖__libc_lock_lock 的保護,可能導致線程并發(fā)環(huán)境下系統(tǒng)性能受限。而改為'+8:00'則不會觸發(fā)系統(tǒng)時區(qū)轉換,使用 MySQL 自身轉換,大大提高了性能。
總結:
讀完本篇文章,你是否對數據庫時區(qū)有了更深刻的認識呢。希望這篇文章對你有所幫助,特別是想了解 MySQL 時區(qū)相關內容時,可以拿來多讀讀。如果你遇到過其他時區(qū)相關問題,歡迎留言討論。
以上就是mysql解決時區(qū)相關問題的詳細內容,更多關于mysql時區(qū)相關問題的資料請關注腳本之家其它相關文章!
標簽:臺灣 鎮(zhèn)江 阜新 合肥 日照 貴州 鷹潭 北京
巨人網絡通訊聲明:本文標題《mysql解決時區(qū)相關問題》,本文關鍵詞 mysql,解決,時區(qū),相關,問題,;如發(fā)現(xiàn)本文內容存在版權問題,煩請?zhí)峁┫嚓P信息告之我們,我們將及時溝通與處理。本站內容系統(tǒng)采集于網絡,涉及言論、版權與本站無關。下一篇:MySQL8.0內存相關參數總結