熟悉Redis的同學應該知道,Redis的每個Key都可以設置一個過期時間,當達到過期時間的時候,這個key就會被自動刪除。這就是Redis的過期策略。
在為key設置過期時間需要注意的事項
1、 DEL/SET/GETSET等命令會清除過期時間
在使用 DEL、SET、GETSET 等會覆蓋key對應value的命令操作一個設置了過期時間的key的時候,會導致對應的key的過期時間被清除。
2、INCR/LPUSH/HSET等命令則不會清除過期時間
而在使用 INCR/LPUSH/HSET 這種只是修改一個key的value,而不是覆蓋整個value的命令,則不會清除key的過期時間。INCR:
LPUSH:
3、PERSIST命令會清除過期時間
當使用 PERSIST 命令將一個設置了過期時間的key轉變成一個持久化的key的時候,也會清除過期時間。
4、使用RENAME命令,老key的過期時間將會轉到新key上
在使用例如: RENAME KEY A KEY B 命令將KEYA重命名為KEYB,不管KEYB有沒有設置過期時間,新的key KEYB將會繼承KEY_A的所有特性。
這裡篇幅有限,我就不一一將keya重命名到keyb的各個情況列出來,大家可以在自己電腦上試一下keya設置了過期時間,keyb沒設置過期時間這種情況。
5、使用EXPIRE/PEXPIRE設置的過期時間為負數或者使用EXPIREAT/PEXPIREAT設置過期時間戳為過去的時間會導致key被刪除
EXPIRE:
EXPIREAT:
6、EXPIRE命令可以更新過期時間
對一個已經設置了過期時間的key使用expire命令,可以更新其過期時間。
在Redis2.1.3以下的版本中,使用expire命令更新一個已經設置了過期時間的key的過期時間會失敗。並且對一個設置了過期時間的key使用LPUSH/HSET等命令修改其value的時候,會導致Redis刪除該key。
Redis的過期策略
那你有沒有想過一個問題,Redis裡面如果有大量的key,怎樣才能高效的找出過期的key並將其刪除呢,難道是遍歷每一個key嗎?假如同一時期過期的key非常多,Redis會不會因為一直處理過期事件,而導致讀寫指令的卡頓。
這裡說明一下,Redis是單線程的,所以一些耗時的操作會導致Redis卡頓,比如當Redis數據量特別大的時候,使用keys * 命令列出所有的key。
實際上Redis使用 懶惰刪除+定期刪除 相結合的方式處理過期的key。
懶惰刪除
所謂 懶惰刪除 就是在客戶端訪問該key的時候,redis會對key的過期時間進行檢查,如果過期了就立即刪除。
這種方式看似很完美,在訪問的時候檢查key的過期時間,不會佔用太多的額外CPU資源。但是如果一個key已經過期了,如果長時間沒有被訪問,那麼這個key就會一直存留在內存之中,嚴重消耗了內存資源。
定期刪除
定期刪除的原理是,Redis會將所有設置了過期時間的key放入一個字典中,然後每隔一段時間從字典中隨機一些key檢查過期時間並刪除已過期的key。
Redis默認每秒進行10次過期掃描:
1、從過期字典中隨機20個key
2、刪除這20個key中已過期的
3、如果超過25%的key過期,則重複第一步
同時,為了保證不出現循環過度的情況,Redis還設置了掃描的時間上限,默認不會超過25ms。