一扔茅、時區(qū)與時間:Mysql已旧,JDBC,JVM

0. Mysql的TimeStamp召娜、DateTime

1)TimeStamp
  • 時間范圍:'1970-01-01 00:00:01' UTC to '2038-01-19 03:14:07' UTC.
  • 以整數(shù)格式存儲运褪,代表自epoch后的秒數(shù)。
  • 存儲大小為4個字節(jié)(不考慮小數(shù)部分)玖瘸,即4*8=32位秸讹,除去最高位的符號位,所能表達的最大數(shù)為2^31=2,147,483,648雅倒;
2)DateTime
  • 時間范圍:'1000-01-01 00:00:00' to '9999-12-31 23:59:59'
  • 存儲占用5個字節(jié)
 1 bit  sign           (1= non-negative, 0= negative)
17 bits year*13+month  (year 0-9999, month 0-12)
 5 bits day            (0-31)
 5 bits hour           (0-23)
 6 bits minute         (0-59)
 6 bits second         (0-59)
---------------------------
40 bits = 5 bytes   //來源:mysql官網(wǎng)
3)時區(qū)對DateTime和TimeStamp的影響

以下為mysql官網(wǎng)11.2.2的測試案例
說明:unix_timestamp是當前時區(qū)相對于1970年的秒數(shù)璃诀;

mysql> CREATE TABLE ts (
    ->     id INTEGER NOT NULL AUTO_INCREMENT PRIMARY KEY,
    ->     col TIMESTAMP NOT NULL
    -> ) AUTO_INCREMENT = 1;

mysql> CREATE TABLE dt (
    ->     id INT NOT NULL AUTO_INCREMENT PRIMARY KEY,
    ->     col DATETIME NOT NULL
    -> ) AUTO_INCREMENT = 1;
 
mysql> SET @@time_zone = 'SYSTEM';

mysql> INSERT INTO ts (col) VALUES ('2020-01-01 10:10:10'),
    ->     ('2020-01-01 10:10:10+05:30'), ('2020-01-01 10:10:10-08:00');
 
mysql> SET @@time_zone = '+00:00';

mysql> INSERT INTO ts (col) VALUES ('2020-01-01 10:10:10'),
    ->     ('2020-01-01 10:10:10+05:30'), ('2020-01-01 10:10:10-08:00');
 
mysql> SET @@time_zone = 'SYSTEM';

mysql> INSERT INTO dt (col) VALUES ('2020-01-01 10:10:10'),
    ->     ('2020-01-01 10:10:10+05:30'), ('2020-01-01 10:10:10-08:00');
 
mysql> SET @@time_zone = '+00:00';

mysql> INSERT INTO dt (col) VALUES ('2020-01-01 10:10:10'),
    ->     ('2020-01-01 10:10:10+05:30'), ('2020-01-01 10:10:10-08:00');
 
mysql> SET @@time_zone = 'SYSTEM';

mysql> SELECT @@system_time_zone;
+--------------------+
| @@system_time_zone |
+--------------------+
| EST                |
+--------------------+

mysql> SELECT col, UNIX_TIMESTAMP(col) FROM dt ORDER BY id;
+---------------------+---------------------+
| col                 | UNIX_TIMESTAMP(col) |
+---------------------+---------------------+
| 2020-01-01 10:10:10 |          1577891410 |
| 2019-12-31 23:40:10 |          1577853610 |
| 2020-01-01 13:10:10 |          1577902210 |
| 2020-01-01 10:10:10 |          1577891410 |
| 2020-01-01 04:40:10 |          1577871610 |
| 2020-01-01 18:10:10 |          1577920210 |
+---------------------+---------------------+

mysql> SELECT col, UNIX_TIMESTAMP(col) FROM ts ORDER BY id;
+---------------------+---------------------+
| col                 | UNIX_TIMESTAMP(col) |
+---------------------+---------------------+
| 2020-01-01 10:10:10 |          1577891410 |
| 2019-12-31 23:40:10 |          1577853610 |
| 2020-01-01 13:10:10 |          1577902210 |
| 2020-01-01 05:10:10 |          1577873410 |
| 2019-12-31 23:40:10 |          1577853610 |
| 2020-01-01 13:10:10 |          1577902210 |
+---------------------+---------------------+

解釋說明:
a)若不指定時區(qū)偏移

  • DateTime與時區(qū)無關(guān),存儲和查詢的操作不會做時區(qū)轉(zhuǎn)換蔑匣,即查詢的值等于存儲時的字符串表示的時間
  • TimeStamp與時區(qū)有關(guān)劣欢,存儲時根據(jù)當前時區(qū)轉(zhuǎn)為相對GMT的時間戳,查詢時根據(jù)當前時區(qū)解析時間戳為當前時區(qū)的時間殖演。

b)若指定時區(qū)偏移(since mysql 8.0.19)

  • DateTime存儲時把指定時區(qū)的Date Time轉(zhuǎn)為當前時區(qū)的Date和Time氧秘,然后存儲,查詢時原樣取出趴久,不做時區(qū)轉(zhuǎn)換丸相。即只在存儲時轉(zhuǎn)換。
    例如:設(shè)置當前時區(qū)為System(ESG, -05:00)彼棍,insert時指定DateTime為2020-01-01 10:10:10+05:30灭忠,根據(jù)時區(qū)的東加西減算法,需要減去10:30, 得到2019-12-31 23:40:10-05:00座硕。存儲2019-12-31 23:40:10
  • TimeStamp存儲時根據(jù)指定的時區(qū)轉(zhuǎn)為相對GMT的時間戳弛作,查詢時根據(jù)當前時區(qū)解析時間戳為當前時區(qū)的時間。

1. MysqlWorkbench/Navicat

1)數(shù)據(jù)庫客戶端可以通過命令設(shè)置session時區(qū)华匾;

2. JDBC

1)JDBC的三個配置項:useLegacyDatetimeCode映琳,useTimezone,serverTimezone

useLegacyDatetimeCode: (驅(qū)動8.0已廢棄)
Default: true
Since: 5.1.6
Use code for DATE/TIME/DATETIME/TIMESTAMP handling in result sets and statements that consistently handles time zone conversions from client to server and back again, or use the legacy code for these datatypes that has been in the driver for backwards-compatibility? Setting this property to 'false' voids the effects of "useTimezone," "useJDBCCompliantTimezoneShift," "useGmtMillisForDatetimes," and "useFastDateParsing."

useTimezone: (驅(qū)動8.0已廢棄)
Default: false
Since: 3.0.2
Convert time/date types between client and server time zones (true/false, defaults to 'false')? This is part of the legacy date-time code, thus the property has an effect only when "useLegacyDatetimeCode=true."

serverTimezone:
Since version: 3.0.2
Override detection/mapping of time zone. Used when time zone from server doesn't map to Java time zone

2)JDBC調(diào)解時區(qū)
以timestamp為例,如果數(shù)據(jù)源的url時區(qū)配置為serverTimezone=GMT萨西,則會對讀取到的timestamp進行時區(qū)轉(zhuǎn)換有鹿。若果當前時區(qū)為東八區(qū),則會對時間+8谎脯;

時區(qū)圖

3. 結(jié)論

  • 不考慮SQL中指定時區(qū)葱跋,則mysql在讀取或?qū)懭霑r不會對datetime根據(jù)時區(qū)來做轉(zhuǎn)換;
  • JDBC會調(diào)解時區(qū)源梭,所以要保證mysql服務(wù)器和serverTimezone一致娱俺;

參考資料

1.Mysql 官網(wǎng)

11.2.2 The DATE, DATETIME, and TIMESTAMP Types
10.9 Date and Time Data Type Representation(存儲格式)
5.1.14 MySQL Server Time Zone Support
5.3 Configuration Properties for Connector/J
6.3 Configuration Properties
https://dev.mysql.com/doc/index-connectors.html
函數(shù)unix-timestamp
6.5 Java, JDBC, and MySQL Types

2.其他

https://stackoverflow.com/questions/7605953/how-to-change-mysql-timezone-in-a-database-connection-using-java
How to Set the JVM Time Zone
https://stackoverflow.com/questions/26515700/mysql-jdbc-driver-5-1-33-time-zone-issue
https://stackoverflow.com/questions/930900/how-do-i-set-the-time-zone-of-mysql/16066034
一次JDBC與MySQL因“CST”時區(qū)協(xié)商誤解導(dǎo)致時間差了14或13小時的排錯經(jīng)歷
時間戳(UnixTimestamp)與 《2038年問題》
Java與MySQL時間戳傳遞/存儲/協(xié)調(diào)問題--userLegacyDatetimeCode--userTimezone--serverTimezone

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市废麻,隨后出現(xiàn)的幾起案子荠卷,更是在濱河造成了極大的恐慌,老刑警劉巖脑溢,帶你破解...
    沈念sama閱讀 207,248評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件僵朗,死亡現(xiàn)場離奇詭異,居然都是意外死亡屑彻,警方通過查閱死者的電腦和手機验庙,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,681評論 2 381
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來社牲,“玉大人粪薛,你說我怎么就攤上這事〔簦” “怎么了违寿?”我有些...
    開封第一講書人閱讀 153,443評論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長熟空。 經(jīng)常有香客問我藤巢,道長,這世上最難降的妖魔是什么息罗? 我笑而不...
    開封第一講書人閱讀 55,475評論 1 279
  • 正文 為了忘掉前任掂咒,我火速辦了婚禮,結(jié)果婚禮上迈喉,老公的妹妹穿的比我還像新娘绍刮。我一直安慰自己,他們只是感情好挨摸,可當我...
    茶點故事閱讀 64,458評論 5 374
  • 文/花漫 我一把揭開白布孩革。 她就那樣靜靜地躺著,像睡著了一般得运。 火紅的嫁衣襯著肌膚如雪膝蜈。 梳的紋絲不亂的頭發(fā)上锅移,一...
    開封第一講書人閱讀 49,185評論 1 284
  • 那天,我揣著相機與錄音彬檀,去河邊找鬼帆啃。 笑死,一個胖子當著我的面吹牛窍帝,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播诽偷,決...
    沈念sama閱讀 38,451評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼坤学,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了报慕?” 一聲冷哼從身側(cè)響起深浮,我...
    開封第一講書人閱讀 37,112評論 0 261
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎眠冈,沒想到半個月后飞苇,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,609評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡蜗顽,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,083評論 2 325
  • 正文 我和宋清朗相戀三年布卡,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片雇盖。...
    茶點故事閱讀 38,163評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡忿等,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出崔挖,到底是詐尸還是另有隱情贸街,我是刑警寧澤,帶...
    沈念sama閱讀 33,803評論 4 323
  • 正文 年R本政府宣布狸相,位于F島的核電站薛匪,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏脓鹃。R本人自食惡果不足惜逸尖,卻給世界環(huán)境...
    茶點故事閱讀 39,357評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望将谊。 院中可真熱鬧冷溶,春花似錦、人聲如沸尊浓。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,357評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽栋齿。三九已至苗胀,卻和暖如春襟诸,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背基协。 一陣腳步聲響...
    開封第一講書人閱讀 31,590評論 1 261
  • 我被黑心中介騙來泰國打工歌亲, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人澜驮。 一個月前我還...
    沈念sama閱讀 45,636評論 2 355
  • 正文 我出身青樓陷揪,卻偏偏與公主長得像,于是被迫代替她去往敵國和親杂穷。 傳聞我的和親對象是個殘疾皇子悍缠,可洞房花燭夜當晚...
    茶點故事閱讀 42,925評論 2 344

推薦閱讀更多精彩內(nèi)容