本篇內容介紹了“Redis key設置過期時間需要注意哪些事項”的有關知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠學有所成!
熟悉Redis的同學應該知道,Redis的每個Key都可以設置一個過期時間,當達到過期時間的時候,這個key就會被自動刪除。
在使用DEL、SET、GETSET等會覆蓋key對應value的命令操作一個設置了過期時間的key的時候,會導致對應的key的過期時間被清除。
//設置mykey的過期時間為300s127.0.0.1:6379> set mykey hello ex 300OK//查看過期時間127.0.0.1:6379> ttl mykey(integer) 294//使用set命令覆蓋mykey的內容127.0.0.1:6379> set mykey ollehOK//過期時間被清除127.0.0.1:6379> ttl mykey(integer) -1
而在使用INCR/LPUSH/HSET這種只是修改一個key的value,而不是覆蓋整個value的命令,則不會清除key的過期時間。
INCR:
//設置incr_key的過期時間為300s127.0.0.1:6379> set incr_key 1 ex 300OK127.0.0.1:6379> ttl incr_key(integer) 291//進行自增操作127.0.0.1:6379> incr incr_key(integer) 2127.0.0.1:6379> get incr_key"2"//查詢過期時間,發現過期時間沒有被清除127.0.0.1:6379> ttl incr_key(integer) 277
LPUSH:
//新增一個list類型的key,并添加一個為1的值127.0.0.1:6379> LPUSH list 1(integer) 1//為list設置300s的過期時間127.0.0.1:6379> expire list 300(integer) 1//查看過期時間127.0.0.1:6379> ttl list(integer) 292//往list里面添加值2127.0.0.1:6379> lpush list 2(integer) 2//查看list的所有值127.0.0.1:6379> lrange list 0 11) "2"2) "1"//能看到往list里面添加值并沒有使過期時間清除127.0.0.1:6379> ttl list(integer) 252
當使用PERSIST命令將一個設置了過期時間的key轉變成一個持久化的key的時候,也會清除過期時間。
127.0.0.1:6379> set persist_key haha ex 300OK127.0.0.1:6379> ttl persist_key(integer) 296//將key變為持久化的127.0.0.1:6379> persist persist_key(integer) 1//過期時間被清除127.0.0.1:6379> ttl persist_key(integer) -1
在使用例如:RENAME KEY_A KEY_B命令將KEY_A重命名為KEY_B,不管KEY_B有沒有設置過期時間,新的key KEY_B將會繼承KEY_A的所有特性。
//設置key_a的過期時間為300s127.0.0.1:6379> set key_a value_a ex 300OK//設置key_b的過期時間為600s127.0.0.1:6379> set key_b value_b ex 600OK127.0.0.1:6379> ttl key_a(integer) 279127.0.0.1:6379> ttl key_b(integer) 591//將key_a重命名為key_b127.0.0.1:6379> rename key_a key_bOK//新的key_b繼承了key_a的過期時間127.0.0.1:6379> ttl key_b(integer) 248
這里篇幅有限,我就不一一將key_a重命名到key_b的各個情況列出來,大家可以在自己電腦上試一下key_a設置了過期時間,key_b沒設置過期時間這種情況。
EXPIRE:
127.0.0.1:6379> set key_1 value_1OK127.0.0.1:6379> get key_1"value_1"//設置過期時間為-1127.0.0.1:6379> expire key_1 -1(integer) 1//發現key被刪除127.0.0.1:6379> get key_1(nil)
EXPIREAT:
127.0.0.1:6379> set key_2 value_2OK127.0.0.1:6379> get key_2"value_2"//設置的時間戳為過去的時間127.0.0.1:6379> expireat key_2 10000(integer) 1//key被刪除127.0.0.1:6379> get key_2(nil)
對一個已經設置了過期時間的key使用expire命令,可以更新其過期時間。
//設置key_1的過期時間為100s127.0.0.1:6379> set key_1 value_1 ex 100OK127.0.0.1:6379> ttl key_1(integer) 95//更新key_1的過期時間為300s127.0.0.1:6379> expire key_1 300(integer) 1127.0.0.1:6379> ttl key_1(integer) 295
在Redis2.1.3以下的版本中,使用expire命令更新一個已經設置了過期時間的key的過期時間會失敗。并且對一個設置了過期時間的key使用LPUSH/HSET等命令修改其value的時候,會導致Redis刪除該key。
那你有沒有想過一個問題,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次過期掃描:
從過期字典中隨機20個key
刪除這20個key中已過期的
如果超過25%的key過期,則重復第一步
同時,為了保證不出現循環過度的情況,Redis還設置了掃描的時間上限,默認不會超過25ms.
“Redis key設置過期時間需要注意哪些事項”的內容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業相關的知識可以關注億速云網站,小編將為大家輸出更多高質量的實用文章!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。