這篇文章主要介紹了在Python的Flask框架中使用日期和時間的教程,包括對各個時區(qū)之間轉(zhuǎn)換的一些處理,需要的朋友可以參考下
時間戳的問題
我們的微博應(yīng)用的一個忽略了很久的問題就是日間和日期的顯示。
直到現(xiàn)在,我們在我們的User和Post對象中使用Python它自己的方式來渲染時間對象,但這并不是一個好的解決方案。
考慮下這樣的例子。我正在寫這篇文章,此時正是12月31號下午3:54。我的時區(qū)是PST(或者你們更習(xí)慣的:UTC-8)。 在Python解釋器中運(yùn)行,我得到下面輸出:
>>> from datetime import datetime
>>> now = datetime.now()
>>> print now
2012-12-31 15:54:42.915204
>>> now = datetime.utcnow()
>>> print now
2012-12-31 23:55:13.635874
在我所在的地方,now()方法返回了正確的時間,但是now()調(diào)用返回的時間是UTC單位。
那么,使用哪個更好呢?
如果我們用now(),所有數(shù)據(jù)庫里的時間戳將會與服務(wù)器運(yùn)行的當(dāng)?shù)貢r間一致,這將會產(chǎn)生一些問題。
比如,如果有一天,我們需要將服務(wù)器放到別的地方(不在一個時區(qū)),那么在重啟服務(wù)器之前,數(shù)據(jù)庫里的時間都需要更新到與新地點(diǎn)保持一致。
還會有更為重要的問題。不同時區(qū)的用戶將會很難知道什么時候發(fā)送郵件,如果用戶看到的是PST時區(qū)的時間,他們就很難知道郵件是什么時候發(fā)送的,這就需要用戶根據(jù)這個時間做相應(yīng)的調(diào)整。
很顯然這不是一個好的選擇,這也是我們?yōu)槭裁丛趧?chuàng)建數(shù)據(jù)庫時就使用UTC時區(qū)保存時間戳。
在標(biāo)準(zhǔn)化時間戳為UTC時,解決了移動服務(wù)器的問題。但是他不能解決第二個問題,數(shù)據(jù)和時間在世界上不同地方使用UTC展現(xiàn)給用戶。
假設(shè)一個用戶在PST時區(qū)下午3點(diǎn)發(fā)送了一封郵件,這封郵件立刻顯示在他面前,上面寫著11:00pm,或者更具體點(diǎn)(23:00)。
我寫這個文章的目的也就是讓我們的用戶不再因為數(shù)據(jù)和時間的顯示而困惑。
使用具體的時間戳
通常的解決方法是,每一個用戶都從UTC轉(zhuǎn)化到當(dāng)?shù)氐臅r間。這就需要我們動態(tài)變化,從而使數(shù)據(jù)庫的UTC與之保持一致。
但是我們怎么知道用戶在哪呢?
許多網(wǎng)站都有一個設(shè)置頁面設(shè)置他們的時區(qū)。這就需要我們添加一個新的頁面,并在表單上提供下拉框讓用戶選擇時區(qū),用戶第一次登錄的時候需要設(shè)置時區(qū),并把它作為注冊的一部分。
這是一個正常的解決方法,但是這對于用戶來說有點(diǎn)累贅,用戶需要輸入一條他們已經(jīng)在操作系統(tǒng)中配置過的信息。所以如果我們能抓取到用戶電腦里設(shè)置的時區(qū)那解決問題會變得更有效率。
出于安全因素,瀏覽器不允許我們進(jìn)入用戶操作系統(tǒng)獲取信息。即使它允許,我們也得知道在Windows,Linux,Mac,iOS,Android中從哪兒能獲得到時區(qū),這還不包括其他非主流操作系統(tǒng)。
在瀏覽器中得到用戶的時區(qū),然后通過標(biāo)準(zhǔn)的Javascript API獲取到。在Web 2.0世界中用戶允許Javascript執(zhí)行(很少有網(wǎng)站不使用Javascript),所以通過Javascript獲取用戶時區(qū)是可行的。
我們用Javascript有兩種方式配置可用的時區(qū):
老派的做法:當(dāng)用戶第一次登錄服務(wù)器時讓瀏覽器以某種方式發(fā)送時區(qū)信息給我們。這個可以通過Ajax調(diào)用,或者更簡單的通過meta refresh tag來實現(xiàn)。一旦服務(wù)器知道了時區(qū)信息,它就能保存它在用戶session中,然后調(diào)整所有頁面的時間顯示。
新派的做法:不改變服務(wù)器端的任何東西,但仍然會發(fā)送UTC時間戳到客戶端瀏覽器。轉(zhuǎn)換UTC到本地時間的工作通過Javascript在客戶端執(zhí)行。
兩種方法都是有效的,但第二種更有優(yōu)勢一點(diǎn)。瀏覽器能依照系統(tǒng)本地配置最好滴完成時間轉(zhuǎn)換。像上午/下午 vs 24小時制,日/月/年 vs 月/日/年 還有其他各種文化的格式,這些格式都是瀏覽器可訪問的,但服務(wù)器就不一定了。
如果這些還不夠,那新派的做法還有一個更大的優(yōu)勢,而且別人已經(jīng)為我們做了這件事(moment.js要登場了)!
moment.js簡介
Moment.js 是一個小、免費(fèi)、開源的Javascript庫,它將日期和事件提升到另一個等級。它提供了能想象到的所有的時間日期格式,下面就是一些。
要在我們的應(yīng)用中使用moment.js就需要在我們的模板文件中寫那么一丟丟的Javascript代碼。我們先來通過ISO 8601 時間來創(chuàng)建一個moment對象。例如:通過上面Python例子的UTC時間來創(chuàng)建一個moment對象,就像這樣:
moment("2012-12-31T23:55:13 Z")
一旦對象被創(chuàng)建,它就可以被轉(zhuǎn)化成各種各樣格式類型的string。例如,將一個灰常冗長的時間顯示轉(zhuǎn)換為本地系統(tǒng)的時間:
moment("2012-12-31T23:55:13 Z").format('LLLL');
下面就是轉(zhuǎn)換以后的時間顯示:
Tuesday, January 1 2013 7:55 AM
這兒有更多的例子將同樣的時間戳轉(zhuǎn)化為不同的格式:
201542191024389.jpg (377×175)
這個類庫對轉(zhuǎn)化選項的支持不止這些。除了format()之外,它還提供了fromNow()和calendar()這些更友好的時間戳轉(zhuǎn)化方法:
201542191055278.jpg (187×88)
注意上面所有的例子中服務(wù)器轉(zhuǎn)換相同的UTC時間,而你自己的本地瀏覽器則會轉(zhuǎn)換不同的時間。
最后我們補(bǔ)上漏掉的一點(diǎn)Javascript小技巧,在頁面中顯而易見的是,代碼實際上由moment返回了string類型。最簡單的完成方式是用Javascript的document.write方法:
<script>
document.write(moment("2012-12-31T23:55:13 Z").format('LLLL'));
</script>
通過使用Javascript的document.write是灰常簡單和直接的方式來生成一部分HTML代碼,然而需要注意的是這種方式有一些限制。最需要主義的一點(diǎn)就是document.write方法只能在document被加載時使用,當(dāng)document加載完成后,它便不能修改document了。這個限制的結(jié)果就是當(dāng)通過 Ajax 來加載數(shù)據(jù)時這種解決方案就失效了。
整合moment.js
這兒我們需要做一點(diǎn)點(diǎn)事把moment.js添加到我們的微博客中.
首先,我們需要下載moment.min.js這個庫到/app/static/js這個文件夾中,這樣它就可以作為靜態(tài)文件為客戶端服務(wù)。
然后我們在我們的模板文件(fileapp/templates/base.html)中添加這個庫(moment.min.js)的引用:
<script src="/static/js/moment.min.js"></script>
現(xiàn)在我們可以在模板文件中添加
更多信息請查看IT技術(shù)專欄