Redis持久化機(jī)制RDB的實(shí)現(xiàn)
基礎(chǔ)
了解過(guò)Redis持久化RDB嘛?可不可以解釋一下什么是RDB。
答: RDB
持久化機(jī)制是將內(nèi)存中的數(shù)據(jù)生成快照并持久化到磁盤(pán)的過(guò)程,RDB
可以通過(guò)手動(dòng)或者自動(dòng)的方式實(shí)現(xiàn)持久化。
那RDB觸發(fā)的方式有哪幾種方式知道嗎?
答: 有兩種,分別是手動(dòng)觸發(fā)和自動(dòng)觸發(fā):
手動(dòng)觸發(fā)
首先是save命令了,這個(gè)指令會(huì)直接阻塞當(dāng)前redis服務(wù)器,知道RDB完成了為止,對(duì)于線(xiàn)上生產(chǎn)環(huán)境數(shù)據(jù)的備份,我們非常非常不建議使用這種方式。
127.0.0.1:6379> save OK
接下來(lái)就是bgsave
指令了,bgsave
則是主進(jìn)程fork一個(gè)子進(jìn)程,由子進(jìn)程完成持久化操作,而主進(jìn)程繼續(xù)處理客戶(hù)端的讀寫(xiě)請(qǐng)求,如果我們需要手動(dòng)實(shí)現(xiàn)持久化,非常推薦使用這種方式。
# 從輸出我們就可以看出這種方式會(huì)將持久化的操作放在后臺(tái)執(zhí)行 127.0.0.1:6379> bgsave Background saving started
接下來(lái)就是自動(dòng)觸發(fā)了
自動(dòng)觸發(fā)我們可以通過(guò)配置實(shí)現(xiàn)redis.conf
的save
參數(shù)實(shí)現(xiàn),如下所示,假如我們希望用戶(hù)20s內(nèi)寫(xiě)入3次就進(jìn)行持久化,只需在配置中加一條save 20 3
即可。
save 20 3
需要注意的是save 20 3
的20s是以redis的時(shí)間間隔為主,并不是用戶(hù)第1次寫(xiě)入后的20s內(nèi)再寫(xiě)入兩次進(jìn)行持久化。
哦,那你能不能給我講一下bgsave的工作流程呢?
答: bgsave
的工作流程如下圖所示,整體可以簡(jiǎn)述為:
- 主進(jìn)程fork出一個(gè)子進(jìn)程,這時(shí)候主進(jìn)程會(huì)被阻塞。
- 子進(jìn)程創(chuàng)建完成后,
redis
客戶(hù)端會(huì)輸出Background saving started
,這就意味子進(jìn)程開(kāi)始進(jìn)行持久化操作了。 - 子進(jìn)程持久化完成后,會(huì)生成一個(gè)rdb文件,將本次的rdb文件通過(guò)原子替換的方式將上一次備份的rdb覆蓋。
- 子進(jìn)程發(fā)送信號(hào)通知父進(jìn)程本次任務(wù)完成。
RDB常見(jiàn)的配置參數(shù)有哪些了解嘛?
答: 首先是dbfilename
,它可以指定rdb
的文件名
# The filename where to dump the DB dbfilename dump.rdb
接下來(lái)就是dir,它可以指定rdb文件的持久化的位置,默認(rèn)取redis服務(wù)端的位置。
dir ./
當(dāng)reids
無(wú)法將文件寫(xiě)入磁盤(pán),我們可以講stop-writes-on-bgsave-error
設(shè)置為yes
,直接關(guān)掉redis的寫(xiě)操作,默認(rèn)為yes
stop-writes-on-bgsave-error yes
rdbcompression
開(kāi)啟后,redis
默認(rèn)會(huì)通過(guò)LZF
算法壓縮rdb
文件。這種方式會(huì)消耗CPU
,但是壓縮后的大小遠(yuǎn)遠(yuǎn)小于內(nèi)存,但是帶來(lái)的收益卻遠(yuǎn)遠(yuǎn)大于這點(diǎn)開(kāi)銷(xiāo),通過(guò)壓縮的文件無(wú)論是通過(guò)網(wǎng)絡(luò)發(fā)送到從節(jié)點(diǎn)還是存儲(chǔ)到硬盤(pán)的空間都是非??捎^(guān)的。
rdbcompression yes
rdbchecksum
開(kāi)啟后,在存儲(chǔ)快照后,還可以讓redis使用CRC64算法來(lái)進(jìn)行數(shù)據(jù)校驗(yàn),但是這樣做會(huì)增加大約10%的性能消耗,如果希望獲取到最大的性能提升,可以關(guān)閉此功能。
rdbchecksum yes
說(shuō)了那么多,可不可以給我演示一下RDB的使用方式
答: 沒(méi)問(wèn)題,我們首先需要存點(diǎn)數(shù)據(jù),20s存3個(gè)值
127.0.0.1:6379> set k1 v1 OK 127.0.0.1:6379> set k2 v2 OK 127.0.0.1:6379> set k3 v3 OK
完成后查看是否生成rdb文件,確認(rèn)無(wú)誤后,我們將這個(gè)文件備份,并強(qiáng)制關(guān)閉redis服務(wù)端,模擬斷電的場(chǎng)景
# 重命名rdb文件 [root@iZ8vb7bhe4b8nhhhpavhwpZ sbin]# mv dump.rdb dump.rdb.bak
此時(shí)我們?cè)賳?dòng)redis就會(huì)發(fā)現(xiàn)數(shù)據(jù)為空
127.0.0.1:6379> keys * (empty array)
我們將rdb文件還原,并重啟redis,可以發(fā)現(xiàn)備份數(shù)據(jù)還原了
# 強(qiáng)制關(guān)閉redis [root@iZ8vb7bhe4b8nhhhpavhwpZ sbin]# ps -ef |grep redis |grep -v grep root 8956 1 0 23:22 ? 00:00:00 redis-server 127.0.0.1:6379 [root@iZ8vb7bhe4b8nhhhpavhwpZ sbin]# kill -9 8956 # 還原rdb,并啟動(dòng)redis [root@iZ8vb7bhe4b8nhhhpavhwpZ sbin]# mv dump.rdb.bak dump.rdb [root@iZ8vb7bhe4b8nhhhpavhwpZ sbin]# redis-server /root/redis/redis.conf [root@iZ8vb7bhe4b8nhhhpavhwpZ sbin]# redis-cli # 可以看到之前設(shè)置的數(shù)據(jù)都回來(lái)了 127.0.0.1:6379> keys * 1) "k3" 2) "k2" 3) "k1"
注:當(dāng)我們使用shutdown指令也會(huì)自動(dòng)觸發(fā)bgsave,讀者可以自行測(cè)試。
不錯(cuò),那你可以說(shuō)說(shuō)RDB有哪些優(yōu)缺點(diǎn)嘛?
答: 首先說(shuō)說(shuō)優(yōu)點(diǎn)吧:
- rdb是緊湊壓縮的二進(jìn)制文件,非常實(shí)用與備份或者全景復(fù)制等場(chǎng)景。
- rdb恢復(fù)數(shù)據(jù)效率遠(yuǎn)遠(yuǎn)高于aof
而缺點(diǎn)如下:
- 無(wú)法做到毫秒級(jí)別的實(shí)時(shí)性持久化,盡管我們可以通過(guò)設(shè)置緊湊的save完成持久化,但是頻繁的fork子進(jìn)程進(jìn)行持久化,很可能造成redis主進(jìn)行長(zhǎng)期阻塞。
- 存儲(chǔ)的文件是二進(jìn)制,不夠直觀(guān),可能還存在某些兼容問(wèn)題。
進(jìn)階
生產(chǎn)環(huán)境大內(nèi)存的redis數(shù)據(jù)如何在持久化的時(shí)候保持?jǐn)?shù)據(jù)一致性呢?
答: redis的rdb持久化是基于Copy on write (寫(xiě)時(shí)復(fù)制思想)
,redis
會(huì)fork
一個(gè)子進(jìn)程完成數(shù)據(jù)持久化,再此期間發(fā)生的原數(shù)據(jù)修改或者寫(xiě)入的新數(shù)據(jù)都會(huì)生成一個(gè)數(shù)據(jù)副本存到一個(gè)新的內(nèi)存區(qū)域中,bgsave
子進(jìn)程快照完成后,再將這塊內(nèi)存區(qū)域同步到原來(lái)的內(nèi)存區(qū)域中,等待下一次快照。
這樣做的缺點(diǎn)也很明顯,極端情況下,如果在bgsave
期間主進(jìn)程數(shù)據(jù)都被改了,那么內(nèi)存占用就是原來(lái)的兩倍。
哦,那在進(jìn)行快照操作的這段時(shí)間,如果發(fā)生服務(wù)崩潰怎么辦?
答: 服務(wù)恢復(fù)的數(shù)據(jù)只會(huì)是上一次備份的rdb
文件數(shù)據(jù),因?yàn)?code>bgsave子進(jìn)程只會(huì)將操作成功的文件生成rdb
文件覆蓋上一次備份的文件。
可以每秒做一次快照嗎?
答: emmm,可以倒是可以,但是可能會(huì)有下面這幾個(gè)問(wèn)題:
- 頻繁寫(xiě)入內(nèi)存數(shù)據(jù)會(huì)給磁盤(pán)帶來(lái)很大的壓力,多個(gè)
fork
子進(jìn)程搶占優(yōu)先的磁盤(pán)帶寬,前一個(gè)子進(jìn)程沒(méi)寫(xiě)完,后一個(gè)子進(jìn)程又來(lái)寫(xiě)入。 rdb
持久化每次fork
子進(jìn)程都會(huì)阻塞主進(jìn)程,頻繁fork
很可能導(dǎo)致主進(jìn)程長(zhǎng)期處于阻塞狀態(tài)。
參考文獻(xiàn)
面試必問(wèn)的 Redis:RDB、AOF、混合持久化
到此這篇關(guān)于Redis持久化機(jī)制RDB的實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)Redis持久化機(jī)制RDB內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
控制Redis的hash的field中的過(guò)期時(shí)間
這篇文章主要介紹了控制Redis的hash的field中的過(guò)期時(shí)間問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-01-01RedisTemplate常用操作方法總結(jié)(set、hash、list、string等)
本文主要介紹了RedisTemplate常用操作方法總結(jié),主要包括了6種常用方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-05-05基于Redis實(shí)現(xiàn)抽獎(jiǎng)功能及問(wèn)題小結(jié)
這篇文章主要介紹了基于Redis實(shí)現(xiàn)抽獎(jiǎng)功能,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-08-08Redis 數(shù)據(jù)類(lèi)型Streams詳解
Redis Streams是Redis 5.0新增的數(shù)據(jù)類(lèi)型,提供了一種日志結(jié)構(gòu)化數(shù)據(jù)存儲(chǔ)方式,這種類(lèi)型適合用于構(gòu)建消息隊(duì)列、事件日志和處理時(shí)間序列數(shù)據(jù)的應(yīng)用,本文介紹Redis 數(shù)據(jù)類(lèi)型Streams相關(guān)知識(shí),感興趣的朋友一起看看吧2024-10-10Redis基本數(shù)據(jù)類(lèi)型Set常用操作命令
這篇文章主要為大家介紹了Redis基本數(shù)據(jù)類(lèi)型Set常用操作命令,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-05-05