国产成人精品久久免费动漫-国产成人精品天堂-国产成人精品区在线观看-国产成人精品日本-a级毛片无码免费真人-a级毛片毛片免费观看久潮喷

您的位置:首頁技術(shù)文章
文章詳情頁

MySQL timestamp與時(shí)區(qū)問題的解決

瀏覽:41日期:2023-08-28 20:27:10
目錄MySQL 用什么類型存日期時(shí)間?JDBC 驅(qū)動(dòng)和 MySQL 的時(shí)區(qū)問題MySQL ClientMySQL Servertimestamp 類型與時(shí)區(qū)的影響總結(jié)參考文獻(xiàn)

新項(xiàng)目可能存在國際化的問題,所以花了點(diǎn)時(shí)間了解了下 MySQL 和 JDBC 驅(qū)動(dòng)相關(guān)的時(shí)間問題。看了好多篇博客發(fā)現(xiàn),不少人理解的都是錯(cuò)誤的,所以結(jié)合官方的文檔,重新梳理了一下。

大家都知道在 MySQL 中有兩個(gè)專門用來存取日期時(shí)間的類型,timestamp 和 datetime。大家總是說 datetime 不包含時(shí)區(qū)信息,timestamp 存的 utc 時(shí)間戳更適合國際化場景下的本地時(shí)間轉(zhuǎn)換。經(jīng)過了解,上述說法對,但不完全對。

版本信息:

MySQL5.7mysql-connector-java 8.0.31MySQL 用什么類型存日期時(shí)間?

目前可以存日期時(shí)間的數(shù)據(jù)類型主要分兩類:

bigint:直接將 utc 時(shí)間戳存到 int 類型的字段中。后期根據(jù)用戶本地時(shí)區(qū)進(jìn)行轉(zhuǎn)換。日期時(shí)間類型:timestamp:MySQL 官方定義的時(shí)間戳,內(nèi)部使用 utc 時(shí)間戳存儲,但查詢時(shí)返回的結(jié)果會(huì)隨著 session time_zone 的變化而變化。datetime:只存日期時(shí)間的值,不包含時(shí)區(qū)信息。

那么我們應(yīng)該選擇哪個(gè)?先說結(jié)論在分析,根據(jù)需求選 bigint 或 datetime。

類型優(yōu)點(diǎn)缺點(diǎn)bigint1. 直觀,不需要考慮時(shí)區(qū)2. 國際化場景直接根據(jù)本地時(shí)區(qū)轉(zhuǎn)化3. 存儲和查詢效率高1. 數(shù)據(jù)庫直接查看時(shí)可讀性差,需要轉(zhuǎn)化timestamp無1. 查詢結(jié)果收到 session time_zone 的影響,容易出錯(cuò)2. 最大時(shí)間為 2038 年datetime1. 可讀性好2. MySQL 自帶各種操作函數(shù),便于查詢3. 存儲日期范圍大,到 9999 年1. 存儲空間略大2. 查詢效率較前兩者低

所以在我看來,如果是后臺管理類的系統(tǒng),更適合使用 datetime,因?yàn)閷π阅芤鬀]那么高,并且查詢方便;其他的可以考慮 bigint。

其中最不建議使用的是 timestamp,因?yàn)檫@個(gè)會(huì)隨著 session time_zone 的變化而變化的,如果開發(fā)者對時(shí)區(qū)理解不深,當(dāng)服務(wù)器或、MySQL 或者 JVM 時(shí)區(qū)發(fā)生變化時(shí),很容易出現(xiàn)問題。

JDBC 驅(qū)動(dòng)和 MySQL 的時(shí)區(qū)問題

剛剛講了 MySQL 的 timestamp 類型的查詢結(jié)果會(huì)受到 session time_zone 的變化而變化,那么這節(jié)我們就來講講時(shí)區(qū)的問題。

那么從 Springboot Application 到 MySQL,哪些地方存在時(shí)區(qū)的配置呢?自然是通信的兩端,MySQL 服務(wù)端和 MySQL 客戶端。

MySQL Client

這里的 Client 是一個(gè)籠統(tǒng)的概念,涉及 client 所在機(jī)器 os 的時(shí)區(qū)、對應(yīng)應(yīng)用程序的時(shí)區(qū)(eg: Java 應(yīng)用就是 JVM 時(shí)區(qū))、JDBC 驅(qū)動(dòng)的時(shí)區(qū)。

機(jī)器 OS 時(shí)區(qū)root@T630-03:/# cat /etc/timezoneAsia/Shanghairoot@T630-03:/# date2023 年 06 月 21 日 星期三 11:20:28 CST

OS 時(shí)區(qū)影響的是 OS 自身的日期時(shí)間命令結(jié)果的返回,同時(shí)還是影響部分軟件,因?yàn)橛行┸浖?huì)默認(rèn)使用 OS 時(shí)區(qū)作為軟件時(shí)區(qū)。

應(yīng)用程序時(shí)區(qū)

這里我們 Spring Boot 應(yīng)用程序?yàn)槔瑢?yīng)的就是 JVM 時(shí)區(qū),它默認(rèn)使用的系統(tǒng)時(shí)區(qū),可通過輸出 ZoneId.*systemDefault*() 來查看。可通過 TimeZone.setDefault(TimeZone.getTimeZone("America/New_York")); 來配置。主要涉及在 Java 中操作日期和時(shí)間的相關(guān)類。比如 LocalDateTime.now() ,假設(shè)北京時(shí)間為 20:00,如果將 TimeZone 設(shè)置為 Asia/Tokyo,則會(huì)返回 21:00。

JDBC 驅(qū)動(dòng)時(shí)區(qū)

在 Springboot 應(yīng)用程序中,我們通過 JDBC 驅(qū)動(dòng)來連接到數(shù)據(jù)庫,在連接的過程中,我們需要配置 jdbcUrl 來指定 JDBC 時(shí)區(qū)。

在 8.0 的驅(qū)動(dòng)中,是通過配置以下字段指定的:

這兩個(gè)參數(shù)指定了,通過 JDBC 驅(qū)動(dòng)連接到 MySQL 的 session time_zone 為 Asia/Shanghai。那么通過 JDBC 驅(qū)動(dòng)去調(diào)用 SQL 中日期時(shí)間相關(guān)的方法都會(huì)以此時(shí)區(qū)為準(zhǔn)。

connectionTimeZone=Asia/ShanghaiforceConnectionTimeZoneToSession=trueMySQL Server

這里的 Server 指的是數(shù)據(jù)庫,涉及數(shù)據(jù)庫所在機(jī)器 os 的時(shí)區(qū)、MySQL 的時(shí)區(qū)。

機(jī)器 OS 時(shí)區(qū)

同上,但 MySQL 默認(rèn)使用的是 OS 時(shí)區(qū)

MySQL 會(huì)話時(shí)區(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 有一個(gè)默認(rèn)的時(shí)區(qū)配置,如上,是使用 OS 時(shí)區(qū)作為 MySQL 默認(rèn)時(shí)區(qū)的。MySQL 其實(shí)本身是沒有所謂的時(shí)區(qū)的,我們所說的都是它的 session time_zone,具體的:

MySQL 存在一個(gè)默認(rèn)時(shí)區(qū),一般是 SYSTEM。可通過配置文件或 SQL 進(jìn)行修改。Client 連接到 MySQL 時(shí)若不指定 session time_zone,默認(rèn)使用 MySQL Server 的時(shí)區(qū)。也可通過 JdbcUrl 在連接的時(shí)候指定或通過 SET time_zone ='UTC'; 指定。Client 在指定了 session time_zone 后,所有的日期時(shí)間查詢均會(huì)按照所在時(shí)區(qū)返回結(jié)果。timestamp 類型與時(shí)區(qū)的影響

timestamp 在 MySQL 中其實(shí)保存就是 UTC 時(shí)間戳,當(dāng) Client 配置不同的會(huì)話時(shí)區(qū)后,會(huì)進(jìn)行轉(zhuǎn)換顯示。

下面舉幾個(gè)最直觀的例子。

表 time_test,有一個(gè) timestamp 類型的字段 client_time。

直接在 MySQL 控制臺操作

SET time_zone = 'Asia/Shanghai';# 2023-06-20 08:00:00insert into time_test(client_time) values (current_timestamp());select * from time_test ;SET time_zone = 'UTC';# 2023-06-20 00:00:00select * from time_test ;# 2023-06-20 00:01:00insert into time_test(client_time) values (current_timestamp());select * from time_test ;SET time_zone = 'Asia/Shanghai';# 2023-06-20 08:01:00select * from time_test ;

? 結(jié)果應(yīng)該很好理解。按照當(dāng)前時(shí)區(qū)插入數(shù)據(jù)后,查詢結(jié)果是當(dāng)前時(shí)區(qū)的日期,更換時(shí)區(qū)后,會(huì)進(jìn)行相應(yīng)的偏移,MySQL 會(huì)自動(dòng)進(jìn)行轉(zhuǎn)換。

通過 JDBC 驅(qū)動(dòng)操作

? 通過 SpringBoot 調(diào)用 JDBC 驅(qū)動(dòng)查詢,雖然中間涉及到了多個(gè)時(shí)區(qū)概念,但其實(shí)轉(zhuǎn)換過程也很簡單。

JVM 時(shí)區(qū)JVM 日期(假設(shè)是 LocalDateTime.now() 返回)JDBC 驅(qū)動(dòng)時(shí)區(qū)插入后返回的查詢結(jié)果Asia/Shanghai2023-06-20 08:00:00Asia/Shanghai2023-06-20 08:00:00Asia/Shanghai2023-06-20 08:00:00UTC2023-06-20 00:00:00Asia/Tokyo2023-06-20 09:00:00Asia/Shanghai2023-06-20 08:00:00Asia/Tokyo2023-06-20 09:00:00UTC2023-06-20 01:00:00

說白了,只與 JDBC 驅(qū)動(dòng)的時(shí)區(qū)有關(guān)。

當(dāng)我們做國際化項(xiàng)目時(shí),只需要保持 JVM 時(shí)區(qū)和 JDBC 驅(qū)動(dòng)時(shí)區(qū)一致,均為 Asia/Shanghai。其他用戶,只需要根據(jù)設(shè)置的本地時(shí)區(qū)進(jìn)行轉(zhuǎn)換即可。

總結(jié)

一番了解下來,最易用的其實(shí)還是bigint和datetime這兩個(gè)時(shí)區(qū)無關(guān)的類型,時(shí)區(qū)相關(guān)的操作直接由我們自己控制最理想。并且也沒有timestamp的時(shí)間限制。總結(jié)下:

優(yōu)先使用bigint和datetime,國際化在代碼層面做。要使用timestamp,需要了解清楚各種時(shí)區(qū)的信息并做好配置。防止因?yàn)闄C(jī)器時(shí)區(qū)改變、數(shù)據(jù)庫時(shí)區(qū)改變而影響查詢的結(jié)果。參考文獻(xiàn)6.3.11 Datetime types processing6.6.1 Preserving Time InstantsSupport for Date-Time Types in Connector/J 8.0

到此這篇關(guān)于MySQL timestamp與時(shí)區(qū)問題的解決的文章就介紹到這了,更多相關(guān)MySQL timestamp與時(shí)區(qū)內(nèi)容請搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標(biāo)簽: MySQL 數(shù)據(jù)庫
相關(guān)文章:
主站蜘蛛池模板: 欧美日本综合一区二区三区 | 欧美videos娇小 | 成人男女视频 | 亚洲自偷 | 免费韩国美女爽快一级毛片 | 俄罗斯美女在线观看一区 | 手机看片久久高清国产日韩 | 免费看一级 | 中国老妇另类xxxx | 香港a毛片免费全部播放 | 国产uv1区二区三区 国产va免费精品高清在线观看 | 香蕉久久a毛片 | 久草视频2 | 女人张开腿男人捅 | 欧美色欧美亚洲高清在线视频 | 日本美女一区二区三区 | 91精品一区二区综合在线 | 真实一级一级一片免费视频 | 国产午夜在线观看视频播放 | 91理论片午午伦夜理片久久 | 一级做a爰片毛片 | 免费v片视频在线观看视频 免费v片在线观看 | 国产三级午夜理伦三级 | 日韩在线观看中文字幕 | 日韩精品一区二区三区视频 | 香港台湾经典三级a视频 | 国产大片一区 | 久久久精品一区二区三区 | 久久久国产精品免费 | 99爱精品视频 | 国产高清国产专区国产精品 | 精品国产一区在线观看 | 亚洲欧美小视频 | 99视频在线精品 | 日韩欧美在线观看 | 国产精品无圣光一区二区 | 在线亚洲一区二区 | 亚洲国产成人精品久久 | 婷婷色综合久久五月亚洲 | 久久视频6免费观看视频精品 | 超级乱淫视频aⅴ播放视频 超级碰碰碰在线观看 |