中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Python datetime 如何處理時區信息

發布時間:2020-10-19 16:21:53 來源:腳本之家 閱讀:372 作者:Yanbin 欄目:開發技術

在 Python 常用日期處理 -- 內置模塊 datetime 探討了 Python 如何使用 datetime, 如果是一個跨時區的應用(Web 應用都是),就不能只存儲一個時間而不帶時區,如此,全球用戶將會看到一個相同的時間字符串,白天黑夜就錯亂了。比說用戶信息的更新時間存儲為 2020-07-07 13:46:08, 上海的用戶和芝加哥的用戶看到的是同一個時間字符串,實質上卻相差好多個小時。

我們可以這么做,在服務端只存儲一個 Timestamp 長整型值或 UTC 時間,Timestamp 是無關乎時區的,它總是相對于一個 UTC 時間的偏移值; 然后由客戶端根據本地時區來顯示當地時間。不過在服務端存儲為 Timestamp 或 UTC 可讀性就不強了,打開文件看到 Timestamp 整形值,大腦是無法直接轉換為日期,UTC 時間略好一些。

另一種做法可在服務端存儲為開發者便于理解的帶時區的時間,如 2020-07-07T13:46:08.342+08:00, 客戶獲得該時間,因為帶有時區信息也就能轉換為客戶端本地時間。

客戶端請求時還可以把本地的時區信息傳送給服務端,由服務端轉換為相應的本地時間發送給客戶端,但 HTTP 頭信息默認不帶時區信息,客戶端必須主動發送它。

本人傾向于在服務端存為帶時區的時間,2020-07-07T13:46:08.342+08:00 是一個標準的存儲格式(ISO_OFFSET_DATE_TIME),客戶端收到它再轉換本地時間,JavaScript 一個很好的組件 moment 處理時間。

探索 Python 對時區的處理

Python 內置組件不能像 Java 的 ZoneId.of("Asia/Shanghai") 直接以時區名獲得 Zone,而需要知道與標準時區的偏移,如上海是東八區,在 Python 中要用 timezone(timedelta(hours=+8)).

那么來看 Python 中輸出帶時區信息,以下是一些應用 Pytho 時區 timezone 的例子

from datetime import datetime, timezone, timedelta
 
tz = timezone(timedelta(hours=+8))
 
fmt = '%Y-%m-%dT%H:%M:%S.%f%z'
zoned_time1 = datetime.today().astimezone(tz)
print(1, zoned_time1.strftime(fmt))  # 2020-07-08T04:30:26.221450+0800
 
zoned_time2 = datetime.now(tz)
print(2, zoned_time2.strftime(fmt))  # 2020-07-08T04:30:26.221543+0800
 
zoned_time3 = datetime.utcnow()
print(3, zoned_time3.isoformat())   # 2020-07-07T20:30:26.221848
 
print(4, zoned_time2.strftime('%Y-%m-%dT%H:%M:%S.%f%Z')) # 2020-07-08T04:30:26.221543UTC+08:00
 
timestamp = datetime.today().timestamp()
print(5, timestamp)         # 1594153826.221895
print(6, datetime.fromtimestamp(timestamp, tz=tz)) # 2020-07-08 04:30:26.221895+08:00
 
zoned_time4 = datetime(2020, 7, 8, 4, 23, 53, 112, tzinfo=tz)
print(7, zoned_time4.isoformat())  # 2020-07-08T04:23:53.000112+08:00
 
print(8, zoned_time2.isoformat())  # 2020-07-08T04:30:26.221543+08:00

輸出為, 已加到上面源代碼中

1 2020-07-08T04:30:26.221450+0800
2 2020-07-08T04:30:26.221543+0800
3 2020-07-07T20:30:26.221848
4 2020-07-08T04:30:26.221543UTC+08:00
5 1594153826.221895
6 2020-07-08 04:30:26.221895+08:00
7 2020-07-08T04:23:53.000112+08:00
8 2020-07-08T04:30:26.221543+08:00

時間字符串中要帶有時區信息,首先時間要轉換為帶時區的,如用

datetime.astimezone(tz)          # 已有時間轉換為帶時區的
datetime.fromtimestamp(timestamp, tz=tz) # 從 timestamp 構建 datetime 時加上時區

找到 Python 輸出標準格式的方法

從上面的輸出結果看第 8 行 2020-07-08T04:30:26.221543+08:00 接近于 Java 的 ISO_OFFSET_DATE_TIME 格式,只是毫秒段 Python 用了 6 位數字,參考 strftime-strptime-behavior 的 Python datetime 格式字符串定義找不到如何把毫秒段收縮為 3 位。

不過注意到 datetime.isoformat() 方法還有一個 timespec 可用,執行下面的代碼

from datetime import datetime, timezone, timedelta
 
tz = timezone(timedelta(hours=+8))
now = datetime.now(tz)
print(now.isoformat(timespec='milliseconds'))

輸出為

2020-07-08T04:41:10.793+08:00

這正式我們想要的。還不僅僅是,繼續往下讀,我們還需要讓 Python 支持夏令時,否則對于芝加哥時間夏天和冬天看到的都是 -5, 那是不對的。

pytz 組件構建時區

Python 也有一個通過時區名稱獲得 timezone 的組件,那就是 pytz - Python TimeZone

$ pip install pytz

測試 pytz

from datetime import datetime
from pytz import timezone
 
tz_shanghai = timezone('Asia/Shanghai')
tz_chicago = timezone('America/Chicago')
 
print(datetime.now(tz=tz_shanghai).isoformat(timespec='milliseconds'))
print(datetime.now(tz=tz_chicago).isoformat(timespec='milliseconds'))

輸出為

2020-07-08T04:55:29.699+08:00
2020-07-07T15:55:29.699-05:00

關于夏令時與冬令時

國內實行夏令時制還是很多年前的事了,80 后初期生人或許還有些印象,就是下午放了學走到街上就能看到《新聞聯播》。為了達成一切形式的統一,我們不再實行夏令時制了,避免了造成可能的分裂。但其他國家仍然有下令時,這會造成同一個地方在一年中產生兩個時區。

例如芝加哥,在夏季時 timezone 是 -05:00, 冬季時是 -06:00.

現在就來看一下 Python 是否能正確的處理夏令時(Date Saving Time)與冬令時(Night Saving Time)。回看上面代碼是在 7 月份執行的結果,此時如果把本地時間改為 12 月份,再看輸出

2020-12-08T06:10:27.862+08:00
2020-12-07T16:10:27.862-06:00

上海的時區仍然為 +08:00, 而芝加哥的時區變成了 -06:00

Python 本身不支持對時令的支持,Python 只知道與 UTC 標準時區的偏移,timezone(timedelta(hours=-5),夏天冬天它的偏移都是 -5,實現夏令冬令時是由 pytz 達成的,同樣的 tz = timezone('America/Chiago')

夏天的結果是 Python 的 timezone(timedelta(hours=-5))
冬天的結果是 Python 的 timezone(timedelta(hours=-6))

對比 Java 對時區的處理

不妨看下隔壁 Java 是如何對時區處理的,分別測試了新舊時間 API

Date today = new Date();
System.out.println(new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss.SSSXXX").format(today));
 
ZonedDateTime now = ZonedDateTime.now(ZoneId.of("Asia/Shanghai"));
System.out.println(now.format(DateTimeFormatter.ISO_OFFSET_DATE_TIME));

2020-07-07T14:53:55.017-05:00
2020-07-08T03:53:55.031+08:00

小結一下

時間用 Timestamp(長整形值) 或統一的 UTC 時間存儲和傳輸,在顯示時轉換為本地時間,但存儲介質上可讀性差
用 timezone(timedelta(hours=-5)) 應用時區來存儲,可讀性增強,但會有夏/冬令時間誤差問題
用 pytz 的 timezone('America/Chicago') 由時區名來構造 timezone 很好的解決了時區和夏/冬令時的問題

以上就是Python datetime 如何處理時區信息的詳細內容,更多關于Python datetime 處理時區信息的資料請關注億速云其它相關文章!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

新郑市| 高雄县| 革吉县| 西林县| 商城县| 安达市| 剑阁县| 武胜县| 双城市| 高台县| 罗山县| 如皋市| 武邑县| 蒲江县| 尼玛县| 天祝| 汉寿县| 乌兰浩特市| 镇巴县| 嘉义市| 柏乡县| 红河县| 奉节县| 竹溪县| 禹城市| 会同县| 定兴县| 读书| 边坝县| 平江县| 栖霞市| 阳曲县| 冷水江市| 濉溪县| 灵川县| 乐陵市| 涟水县| 塔城市| 阿鲁科尔沁旗| 聂拉木县| 巴塘县|