MongoDB中存在一種索引,叫做TTL索引(time-to-live index,具有生命周期的索引)夕膀,這種索引允許為每一個(gè)文檔設(shè)置一個(gè)超時(shí)時(shí)間羽利。一個(gè)文檔達(dá)到預(yù)設(shè)置的老化程度后就會(huì)被刪除变丧。
數(shù)據(jù)到期對(duì)于某些類型的信息非常有用,例如機(jī)器生成的事件數(shù)據(jù)把将,日志和會(huì)話信息没酣,這些信息只需要在數(shù)據(jù)庫中保存有限的時(shí)間王财。
在createIndex中指定expireAfterSeconds選項(xiàng)就可以創(chuàng)建一個(gè)TTL索引:
// 超時(shí)時(shí)間為24小時(shí),默認(rèn)是前臺(tái)運(yùn)行,可以通過background:true設(shè)置為后臺(tái)模式
db.user_session.createIndex({"updated":1},{expireAfterSeconds:60*60*24});
這樣在updated字段上創(chuàng)建了一個(gè)TTL索引裕便。如果一個(gè)文檔的updated字段存在并且它的值是日期類型绒净,當(dāng)服務(wù)器時(shí)間比文檔的updated字段的時(shí)間晚expireAfterSeconds秒時(shí),文檔就會(huì)被刪除偿衰。
db.getCollection('user_session').insert(
{
_id: NumberInt(1),
"updated":new Date(),
username:'lisi'
}
);
mongodb保存時(shí)間使用的UTC時(shí)間挂疆,在查詢出來的結(jié)果的時(shí)候會(huì)轉(zhuǎn)換為GMT時(shí)間,所以你看到保存的時(shí)間和電腦時(shí)間相差8個(gè)小時(shí)(GMT+8)
db.getCollection('user_session').find({updated:{$gt: new Date("2019-07-12 14:00:00")}}) 在查詢的時(shí)候可以使用new Date()直接進(jìn)行時(shí)間的比較下翎,new Date傳入的參數(shù)是GMT時(shí)間
為了防止活躍的會(huì)話被刪除缤言,可以在會(huì)話上有活動(dòng)發(fā)生時(shí)將updated字段的值更新為當(dāng)前時(shí)間。只要updated的時(shí)間距離當(dāng)前時(shí)間達(dá)到24小時(shí)视事。相應(yīng)的文檔就會(huì)被刪除胆萧。
MongoDB的TTL功能依賴于mongodb中的后臺(tái)線程,該線程讀取索引中的日期類型值并從集合中刪除過期的文檔郑口。
MongoDB每分鐘對(duì)TTL索引進(jìn)行一次清理鸳碧,所以不應(yīng)該依賴以秒為單位的時(shí)間保證索引的存活狀態(tài)。而且TTL索引不保證在到期時(shí)立即刪除過期數(shù)據(jù)犬性。文檔到期的時(shí)間與MongoDB從數(shù)據(jù)庫中刪除文檔的時(shí)間之間可能存在延遲瞻离。由于刪除過期文檔的后臺(tái)任務(wù)每60秒運(yùn)行一次。所以乒裆,文檔可能在文檔到期和后臺(tái)任務(wù)運(yùn)行之間的期間保留在集合中套利。
源碼在 https://github.com/mongodb/mongo/blob/master/src/mongo/db/ttl.cpp
mongodb不支持使用createIndex來重新設(shè)置過期時(shí)間,只可以使用collMod命令修改expireAfterSeconds的值:
db.runCommand({collMod:"user_session",index: {name:"updated_1",expireAfterSeconds: 120}});
修改成功后,你會(huì)收到這樣的消息(之前的過期時(shí)間是一分鐘,現(xiàn)在修改為2分鐘)
{
"expireAfterSeconds_old" : 60.0,
"expireAfterSeconds_new" : 120.0,
"ok" : 1.0
}
在一個(gè)給定的集合上可以有多個(gè)TTL索引肉迫,你可以在created和updated字段分別建立ttl索引,但是不能同時(shí)使用兩個(gè)字段建立復(fù)合ttl索引,也不能在同一個(gè)字段上又是創(chuàng)建TTL索引验辞,又是創(chuàng)建普通索引,但是可以像“普通索引”一樣用來優(yōu)化排序和查詢喊衫。