redis?設(shè)置生存和過(guò)期時(shí)間的原理分析
在了解原理前 先來(lái)看使用方法
通過(guò)EXPIRE命令或者PEXPIRE命令,客戶端可以以秒或者毫秒精度為數(shù)據(jù)庫(kù)中的某個(gè)鍵設(shè)置生存時(shí)間,在經(jīng)過(guò)指定的秒數(shù)或者毫秒數(shù)之后,服務(wù)器就會(huì)自動(dòng)刪除生存時(shí)間為0的鍵。
SETEX命令可以在設(shè)置一個(gè)字符串鍵的同時(shí)為鍵設(shè)置過(guò)期時(shí)間(只能用于字符串鍵)

與EXPIRE命令和PEXPIRE命令類似,客戶端可以通過(guò)EXPIREAT命令或PEXPIREAT命令,以秒或者毫秒精度給數(shù)據(jù)庫(kù)中的某個(gè)鍵設(shè)置過(guò)期時(shí)間
過(guò)期時(shí)間是一個(gè)UNIX時(shí)間戳,當(dāng)鍵的過(guò)期時(shí)間來(lái)臨時(shí),服務(wù)器就會(huì)自動(dòng)從數(shù)據(jù)庫(kù)中刪除這個(gè)鍵

TTL命令和PTTL命令接受一個(gè)帶有生存時(shí)間或者過(guò)期時(shí)間的鍵,返回這個(gè)鍵的剩余生存時(shí)間,也就是,返回距離這個(gè)鍵被服務(wù)器自動(dòng)刪除還有多長(zhǎng)時(shí)間

Redis有四個(gè)不同的命令可以用于設(shè)置鍵的生存時(shí)間(鍵可以存在多久)或過(guò)期時(shí)間(鍵什么時(shí)候會(huì)被刪除):
- EXPIRE<key><ttl>命令用于將鍵key的生存時(shí)間設(shè)置為ttl秒。
- PEXPIRE<key><ttl>命令用于將鍵key的生存時(shí)間設(shè)置為ttl毫秒。
- EXPIREAT<key><timestamp>命令用于將鍵key的過(guò)期時(shí)間設(shè)置為timestamp所指定的秒數(shù)時(shí)間戳。
- PEXPIREAT<key><timestamp>命令用于將鍵key的過(guò)期時(shí)間設(shè)置為timestamp所指定的毫秒數(shù)時(shí)間戳。
原理
雖然有多種不同單位和不同形式的設(shè)置命令,但實(shí)際上EXPIRE、PEXPIRE、EXPIREAT三個(gè)命令都是使用PEXPIREAT命令來(lái)實(shí)現(xiàn)的:
無(wú)論客戶端執(zhí)行的是以上四個(gè)命令中的哪一個(gè),經(jīng)過(guò)轉(zhuǎn)換之后,最終的執(zhí)行效果都和執(zhí)行PEXPIREAT命令一樣。

redisDb結(jié)構(gòu)的expires字典保存了數(shù)據(jù)庫(kù)中所有鍵的過(guò)期時(shí)間,我們稱這個(gè)字典為過(guò)期字典
過(guò)期字典的鍵是一個(gè)指針,這個(gè)指針指向鍵空間中的某個(gè)鍵對(duì)象(也即是某個(gè)數(shù)據(jù)庫(kù)鍵)。
過(guò)期字典的值是一個(gè)long long類型的整數(shù),這個(gè)整數(shù)保存了鍵所指向的數(shù)據(jù)庫(kù)鍵的過(guò)期時(shí)間——一個(gè)毫秒精度的UNIX時(shí)間戳。

下圖展示了一個(gè)帶有過(guò)期字典的數(shù)據(jù)庫(kù)例子,在這個(gè)例子中,鍵空間保存了數(shù)據(jù)庫(kù)中的所有鍵值對(duì),而過(guò)期字典則保存了數(shù)據(jù)庫(kù)鍵的過(guò)期時(shí)間。
為了展示方便,圖中的鍵空間和過(guò)期字典中重復(fù)出現(xiàn)了兩次alphabet鍵對(duì)象和book鍵對(duì)象。在實(shí)際中,鍵空間的鍵和過(guò)期字典的鍵都指向同一個(gè)鍵對(duì)象,所以不會(huì)出現(xiàn)任何重復(fù)對(duì)象,也不會(huì)浪費(fèi)任何空間。

圖中的過(guò)期字典保存了兩個(gè)鍵值對(duì):
第一個(gè)鍵值對(duì)的鍵為alphabet鍵對(duì)象,值為1385877600000,這表示數(shù)據(jù)庫(kù)鍵alphabet的過(guò)期時(shí)間為1385877600000(2013年12月1日零時(shí))。
第二個(gè)鍵值對(duì)的鍵為book鍵對(duì)象,值為1388556000000,這表示數(shù)據(jù)庫(kù)鍵book的過(guò)期時(shí)間為1388556000000(2014年1月1日零時(shí))。當(dāng)客戶端執(zhí)行PEXPIREAT命令(或者其他三個(gè)會(huì)轉(zhuǎn)換成PEXPIREAT命令的命令)為一個(gè)數(shù)據(jù)庫(kù)鍵設(shè)置過(guò)期時(shí)間時(shí),服務(wù)器會(huì)在數(shù)據(jù)庫(kù)的過(guò)期字典中關(guān)聯(lián)給定的數(shù)據(jù)庫(kù)鍵和過(guò)期時(shí)間。
在服務(wù)器執(zhí)行以下命令之后
![]()
過(guò)期字典將新增一個(gè)鍵值對(duì),其中鍵為message鍵對(duì)象,而值則為1391234400000(2014年2月1日零時(shí)),如圖

以下是PEXPIREAT命令的偽代碼定義

PERSIST命令可以移除一個(gè)鍵的過(guò)期時(shí)間

PERSIST命令就是PEXPIREAT命令的反操作:PERSIST命令在過(guò)期字典中查找給定的鍵,并解除鍵和值(過(guò)期時(shí)間)在過(guò)期字典中的關(guān)聯(lián)。
過(guò)期鍵的判定
通過(guò)過(guò)期字典,程序可以用以下步驟檢查一個(gè)給定鍵是否過(guò)期:
1)檢查給定鍵是否存在于過(guò)期字典:如果存在,那么取得鍵的過(guò)期時(shí)間。
2)檢查當(dāng)前UNIX時(shí)間戳是否大于鍵的過(guò)期時(shí)間:如果是的話,那么鍵已經(jīng)過(guò)期;否則的話,鍵未過(guò)期。可以用偽代碼來(lái)描述這一過(guò)程:

對(duì)于一個(gè)過(guò)期時(shí)間為1385877600000(2013年12月1日零時(shí))的鍵alphabet來(lái)說(shuō):
如果當(dāng)前時(shí)間為1383282000000(2013年11月1日零時(shí)),那么調(diào)用is_expired(alphabet)將返回False,因?yàn)楫?dāng)前時(shí)間小于alphabet鍵的過(guò)期時(shí)間。
另一方面,如果當(dāng)前時(shí)間為1385964000000(2013年12月2日零時(shí)),那么調(diào)用is_expired(alphabet)將返回True,因?yàn)楫?dāng)前時(shí)間大于alphabet鍵的過(guò)期時(shí)間。
Redis的過(guò)期鍵刪除策略原理
http://www.dhdzp.com/article/260852.htm
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Redis使用布隆過(guò)濾器解決緩存雪崩的問(wèn)題
布隆過(guò)濾器可以幫助我們解決Redis緩存雪崩的問(wèn)題,那什么是布隆過(guò)濾器、布隆過(guò)濾器又是如何使用如何解決緩存雪崩的問(wèn)題的,讓我們帶著這一系列的問(wèn)題去詳細(xì)了解布隆過(guò)濾器,感興趣的小伙伴跟著小編一起來(lái)看看吧2024-02-02
一文詳細(xì)介紹Redis7持久化機(jī)制RDB和AOF
這篇文章主要給大家分享一下Redis的數(shù)據(jù)持久化方式,Reids是一個(gè)高性能的緩存中間件,它的高性能是因?yàn)樗腔趦?nèi)存的,我們知道直接操縱內(nèi)存是比較快的,所以當(dāng)機(jī)器發(fā)生宕機(jī),那么數(shù)據(jù)就會(huì)完全丟失,所以本文詳細(xì)介紹Redis7持久化機(jī)制RDB和AOF2023-07-07

