欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

mysql里CST時(shí)區(qū)的坑及解決

 更新時(shí)間:2023年10月12日 09:50:00   作者:石頭wang  
這篇文章主要介紹了mysql里CST時(shí)區(qū)的坑及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

一、問題簡(jiǎn)述

mysql 里 CST 時(shí)區(qū)是個(gè)非??拥母拍?,因?yàn)樵?mysql 里CST既表示中國(guó)也表示美國(guó)的時(shí)區(qū)。

但是在Jdk代碼里,CST 這個(gè)字符串被理解為Central Standard Time (USA)(GMT-6),這就是造成坑的原因。

解決辦法:

mysql 的 time_zone配置 不要用SYSTEM,因?yàn)橛昧司褪歉S system_time_zone 的值,而 system_time_zone 讀取自數(shù)據(jù)庫(kù)所在宿主機(jī)的操作系統(tǒng),常常是 CST 的值。

解決辦法是將 time_zone 改成例如 +08:00 以免造成誤解。(肯定改mysql啦,你改得了jdk源碼嗎?)。

修改方法見后文

二、查看、修改時(shí)區(qū)

1、怎么查看mysql是什么時(shí)區(qū)?

選一個(gè)即可

方法1:

命令: 連上mysql后命令行執(zhí)行 show variables like '%time_zone%' (Navicat連上后執(zhí)行也行)

結(jié)果:

system_time_zone	CST
time_zone	SYSTEM

看time_zone就行了,SYSTEM表示其值跟隨system_time_zone。

而system_time_zone的CST值,是表示中國(guó)呢,還是美國(guó),是不清楚的,你直接再執(zhí)行 select now() 跟你手機(jī)的時(shí)間對(duì)比一下就清楚了。

為什么會(huì)system_time_zone和time_zone兩個(gè)? 要看哪個(gè)? 不急后面會(huì)解釋

方法2:

命令:連上mysql后命令行執(zhí)行 (Navicat連上后執(zhí)行也行)

  // 分兩次執(zhí)行下面2條命令
  select @@system_time_zone;
  select @@time_zone;

結(jié)果:

  select @@system_time_zone;
  +--------------------+
  | @@system_time_zone |
  +--------------------+
  | CST                |
  +--------------------+
  1 row in set (4.81 sec)
  mysql> select @@time_zone;
  +-------------+
  | @@time_zone |
  +-------------+
  | SYSTEM      |
  +-------------+
  1 row in set (0.04 sec)

方法3:

// 查出全局時(shí)區(qū)和會(huì)話時(shí)區(qū)
select @@GLOBAL.time_zone,@@SESSION.time_zone;

為什么mysql有system_time_zone和time_zone兩個(gè)

https://dev.mysql.com/doc/refman/8.0/en/time-zone-support.html

簡(jiǎn)單說就是system_time_zone就是mysql服務(wù)(即mysqld)啟動(dòng)的時(shí)候讀取數(shù)據(jù)庫(kù)所在宿主機(jī)的時(shí)區(qū),固定下來的值,這個(gè)值之后不再改變(除非重啟mysql服務(wù))。time_zone的值未SYSTEM,意思是它的時(shí)區(qū)跟system_time_zone一樣(注意system_time_zone的值固定下來后,數(shù)據(jù)庫(kù)宿主機(jī)的時(shí)區(qū)再改變,time_zone的值都是不變的,因?yàn)樗歉Ssystem_time_zone變量的,不是實(shí)時(shí)跟隨操作系統(tǒng)的)

存在 “為什么linux時(shí)區(qū)是對(duì)的,但是mysql的時(shí)區(qū)是錯(cuò)的” 的疑惑,因?yàn)閱?dòng)mysql服務(wù)時(shí)如果linux時(shí)區(qū)是錯(cuò)的,把mysql的時(shí)區(qū)也帶錯(cuò)了,只是后來你發(fā)現(xiàn)linux時(shí)區(qū)錯(cuò)了于是改對(duì)了,但是你忘記啟動(dòng)mysql的時(shí)候linux時(shí)區(qū)是錯(cuò)的,而且你忘了是后來你才把linux的時(shí)區(qū)改正確的,所以你有這個(gè)的疑惑。我好像也有過這樣的疑惑

2、如何修改mysql的時(shí)區(qū)

永久修改,改配置文件,重啟mysql服務(wù)也有效

修改 my.cnf 文件,在 [mysqld] 節(jié)下增加 default-time-zone = '+08:00'

沒有這個(gè)文件的話,見后文,讓你的mysql啟動(dòng)的時(shí)候讀取配置文件。

臨時(shí)修改,重啟mysql服務(wù)后丟失

選擇下面之一執(zhí)行即可。

  • 修改會(huì)話級(jí)別定的,關(guān)閉會(huì)話后就會(huì)失效,也僅僅影響當(dāng)前會(huì)話窗口:set time_zone='+08:00';
  • 全局修改,重啟mysql服務(wù)后丟失,對(duì)所有會(huì)話生效:`set global time_zone=’+08:00’;

三、這個(gè)bug 是怎么產(chǎn)生的

下面基于mysql-connector-java-8.0.19 進(jìn)行分析,使用 com.mysql.cj.jdbc.Driver

在獲取java.sql.Connection的過程會(huì)確定時(shí)區(qū),調(diào)用如下方法

com.mysql.cj.protocol.a.NativeProtocol#configureTimezone
/**
    * Configures the client's timezone if required.
    * 
    * @throws CJException
    *             if the timezone the server is configured to use can't be
    *             mapped to a Java timezone.
    */
public void configureTimezone() {
String configuredTimeZoneOnServer = this.serverSession.getServerVariable("time_zone");
if ("SYSTEM".equalsIgnoreCase(configuredTimeZoneOnServer)) {
configuredTimeZoneOnServer = this.serverSession.getServerVariable("system_time_zone");
}
String canonicalTimezone = getPropertySet().getStringProperty(PropertyKey.serverTimezone).getValue();
if (configuredTimeZoneOnServer != null) {
// user can override this with driver properties, so don't detect if that's the case
if (canonicalTimezone == null || StringUtils.isEmptyOrWhitespaceOnly(canonicalTimezone)) {
    try {
    canonicalTimezone = TimeUtil.getCanonicalTimezone(configuredTimeZoneOnServer, getExceptionInterceptor());
    } catch (IllegalArgumentException iae) {
    throw ExceptionFactory.createException(WrongArgumentException.class, iae.getMessage(), getExceptionInterceptor());
    }
}
}
if (canonicalTimezone != null && canonicalTimezone.length() > 0) {
this.serverSession.setServerTimeZone(TimeZone.getTimeZone(canonicalTimezone));
//
// The Calendar class has the behavior of mapping unknown timezones to 'GMT' instead of throwing an exception, so we must check for this...
//
if (!canonicalTimezone.equalsIgnoreCase("GMT") && this.serverSession.getServerTimeZone().getID().equals("GMT")) {
    throw ExceptionFactory.createException(WrongArgumentException.class, Messages.getString("Connection.9", new Object[] { canonicalTimezone }),
                                            getExceptionInterceptor());
}
}
this.serverSession.setDefaultTimeZone(this.serverSession.getServerTimeZone());
}

大概邏輯如下:

  • 從mysql的time_zone讀取值
  • 如果值是SYSTEM則使用system_time_zone的值
  • 如果jdbc url配置了時(shí)區(qū)則使用url里的,如 jdbc:mysql://localhost:3306/test?useSSL=true&serverTimezone=Asia/Shanghai獲取的是字符串配置定的

如果獲取到CST,則因?yàn)樵趈ava里 TimeZone("CST")表示美國(guó)中部時(shí)間,與mysql數(shù)據(jù)庫(kù)用 CST 表示中國(guó)時(shí)區(qū)的含義就不同了,bug就是這么產(chǎn)生的。

四、其他問題

1、CST究竟是美國(guó)還是中國(guó)

其實(shí)CST總共有4個(gè)含義

  • China Standard Time:中國(guó)標(biāo)準(zhǔn)時(shí)間
  • Central Standard Time (USA) :美國(guó)中部時(shí)間(中部就是在國(guó)土的中間的部分,例如芝加哥)
  • Central Standard Time (Australia):澳大利亞中部時(shí)間
  • Cuba Standard Time:古巴標(biāo)準(zhǔn)時(shí)間

就是因?yàn)檫@個(gè)含義不清,所以會(huì)有坑,體現(xiàn)在Java里就是

// 不建議這么創(chuàng)建 TimeZone,這里的CST指的是美國(guó)中部時(shí)間,不是中國(guó),從
TimeZone tz = TimeZone.getTimeZone("CST");
System.out.println(tz);// 可以看到偏移量是offset=-21600000,-21600000微秒=-6小時(shí),所以實(shí)錘這里的CST指美國(guó)
// 建議創(chuàng)建 TimeZone 用 ZoneId,因?yàn)閆oneId 不允許 CST、JST 這種簡(jiǎn)稱,能提前預(yù)防進(jìn)坑,如下
// ZoneId zoneId = ZoneId.of("CST");// 拋異常
ZoneId zoneId = ZoneId.of("GMT+8");// 明確指定,是OK的,或者 "區(qū)域/城市" 的寫法如 Asia/Shanghai
TimeZone tz1 = TimeZone.getTimeZone(zoneId);

體現(xiàn)在Mysql里

你根本不知道m(xù)ysql里的CST是中國(guó)還是美國(guó),查出來都展示一樣

(關(guān)于system_time_zone和time_zone后續(xù)有解釋)

2、你說CST除了表示中國(guó),也表示美國(guó)定的時(shí)區(qū),拿得出證據(jù)嗎?

是的!!! 動(dòng)手做實(shí)驗(yàn)給你看 !!!

使用show variables like '%time_zone%'; 查看mysql的時(shí)區(qū),目前是北京時(shí)間,查出

Variable_name	Value
system_time_zone	CST
time_zone	SYSTEM

當(dāng)把操作系統(tǒng)(mysql的宿主機(jī)器的操作系統(tǒng))的時(shí)區(qū)改成東京時(shí)區(qū),并重啟mysql服務(wù)后(必須重啟服務(wù)),顯示

Variable_name	Value
system_time_zone	JST
time_zone	SYSTEM

最后把操作系統(tǒng)時(shí)區(qū)改成美國(guó)中部時(shí)間(如芝加哥),并重啟mysql服務(wù)后,顯示

Variable_name	Value
system_time_zone	CST
time_zone	SYSTEM

可以看到非???,無論是中國(guó)還是美國(guó),都顯示CST,給人很大的誤解

五、其他

1、mybatis/jdbc 是怎么將日期存入到數(shù)據(jù)庫(kù)的,又是怎么查出來的

(這個(gè)實(shí)在有點(diǎn)復(fù)雜,要考慮mysql所在操作系統(tǒng)的時(shí)區(qū),也考慮跑你的mysql程序的web服務(wù)器的操作系統(tǒng)的時(shí)區(qū)。加又沒時(shí)間繼續(xù)研究了,緩一緩)

2、再次啰嗦補(bǔ)充

mysql里的 system_time_zone 和 time_zone是什么關(guān)系

system_time_zone 是mysql服務(wù)啟動(dòng)時(shí)讀取宿主操作系統(tǒng)的時(shí)區(qū),當(dāng)時(shí)讀取到什么值就固定什么,是那個(gè)時(shí)刻的快照。這個(gè)值只有在被重新指定或重啟mysql服務(wù)的時(shí)候才可能變化。

time_zone就是表示所有連接到此mysql的客戶端(client)的時(shí)區(qū),不管你是什么形式的客戶端,是java通過jdbc連接的程序,還是Navicat這種圖形化的,還是命令行的,都叫客戶端。這個(gè)時(shí)區(qū)是可以被覆蓋的,可以被jdbcUrl中的時(shí)區(qū)覆蓋。

在jdbc的URL里配置了時(shí)區(qū)會(huì)怎么樣?

會(huì)使用這個(gè)時(shí)區(qū),在jdbcUrl里指定的時(shí)區(qū)優(yōu)先級(jí)最高,會(huì)覆蓋其他。

例如 jdbc:mysql://localhost:3306/test?useSSL=true&serverTimezone=Asia/Shanghai

  • time_zone的SYSTEM是什么意思,是不是指 “操作系統(tǒng)” 的時(shí)區(qū)?
  • time_zone的SYSTEM,是指值跟隨 system_time_zone 的值,不是說跟隨操作系統(tǒng)的值,操作系統(tǒng)的時(shí)區(qū)改了,如果 system_time_zone 不改,則 time_zone 也是不會(huì)改的(親自實(shí)驗(yàn)過了!!)

總結(jié)

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • MySQL中datetime和timestamp的區(qū)別及使用詳解

    MySQL中datetime和timestamp的區(qū)別及使用詳解

    這篇文章主要介紹了MySQL中datetime和timestamp的區(qū)別及使用詳解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-11-11
  • win10安裝zip版MySQL8.0.19的教程詳解

    win10安裝zip版MySQL8.0.19的教程詳解

    這篇文章主要介紹了win10安裝zip版MySQL8.0.19的教程詳細(xì),本文分步驟給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-02-02
  • mysql修改數(shù)據(jù)庫(kù)引擎的幾種方法總結(jié)

    mysql修改數(shù)據(jù)庫(kù)引擎的幾種方法總結(jié)

    這篇文章主要給大家介紹了關(guān)于mysql修改數(shù)據(jù)庫(kù)引擎的相關(guān)資料,包括使用ALTERTABLE語(yǔ)句、更改默認(rèn)存儲(chǔ)引擎、使用MySQLWorkbench、導(dǎo)出和導(dǎo)入數(shù)據(jù)以及編寫腳本批量修改,每種方法都有其優(yōu)缺點(diǎn)和適用場(chǎng)景,需要的朋友可以參考下
    2024-11-11
  • 深入mysql基礎(chǔ)知識(shí)的詳解

    深入mysql基礎(chǔ)知識(shí)的詳解

    本篇文章是對(duì)mysql基礎(chǔ)知識(shí)進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
    2013-06-06
  • MySql中表單輸入數(shù)據(jù)出現(xiàn)中文亂碼的解決方法

    MySql中表單輸入數(shù)據(jù)出現(xiàn)中文亂碼的解決方法

    這篇文章主要介紹了MySql中表單輸入數(shù)據(jù)出現(xiàn)中文亂碼的解決方法的相關(guān)資料,需要的朋友可以參考下
    2016-07-07
  • mysql利用覆蓋索引避免回表優(yōu)化查詢

    mysql利用覆蓋索引避免回表優(yōu)化查詢

    這篇文章主要給大家介紹了關(guān)于mysql如何利用覆蓋索引避免回表優(yōu)化查詢的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-02-02
  • 關(guān)于mysql時(shí)間區(qū)間問題淺析

    關(guān)于mysql時(shí)間區(qū)間問題淺析

    在很多地方都使用到了mysql的日期查詢,下面這篇文章主要給大家介紹了關(guān)于mysql時(shí)間區(qū)間問題的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2022-04-04
  • MySQL安裝及初始密碼設(shè)置方式

    MySQL安裝及初始密碼設(shè)置方式

    這篇文章主要介紹了MySQL安裝及初始密碼設(shè)置方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-08-08
  • 完美解決mysql客戶端授權(quán)后連接失敗的問題

    完美解決mysql客戶端授權(quán)后連接失敗的問題

    下面小編就為大家?guī)硪黄昝澜鉀Qmysql客戶端授權(quán)后連接失敗的問題。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2017-03-03
  • linux mysql 數(shù)據(jù)庫(kù)開啟外部訪問設(shè)置指南

    linux mysql 數(shù)據(jù)庫(kù)開啟外部訪問設(shè)置指南

    Linux下設(shè)置MySQL和允許外部機(jī)器訪問,具體目錄是具體情況而定,有的人是安裝了在個(gè)人目錄下,則找到對(duì)應(yīng)的目錄則可以
    2012-11-11

最新評(píng)論