常用的SQL數(shù)據(jù)庫(kù)的數(shù)據(jù)都是存在磁盤(pán)中的,雖然在數(shù)據(jù)庫(kù)底層也做了對(duì)應(yīng)的緩存來(lái)減少數(shù)據(jù)庫(kù)的IO壓力,但由于數(shù)據(jù)庫(kù)的緩存一般是針對(duì)查詢(xún)的內(nèi)容,而且粒度也比較小,一般只有表中的數(shù)據(jù)沒(méi)有發(fā)生變動(dòng)的時(shí)候,數(shù)據(jù)庫(kù)的緩存才會(huì)產(chǎn)生作用,但這并不能減少業(yè)務(wù)邏輯對(duì)數(shù)據(jù)庫(kù)的增刪改操作的IO壓力,因此緩存技術(shù)應(yīng)運(yùn)而生,該技術(shù)實(shí)現(xiàn)了對(duì)熱點(diǎn)數(shù)據(jù)的高速緩存,可以大大緩解后端數(shù)據(jù)庫(kù)的壓力。
創(chuàng)新互聯(lián)公司始終堅(jiān)持【策劃先行,效果至上】的經(jīng)營(yíng)理念,通過(guò)多達(dá)10多年累計(jì)超上千家客戶(hù)的網(wǎng)站建設(shè)總結(jié)了一套系統(tǒng)有效的全網(wǎng)營(yíng)銷(xiāo)推廣解決方案,現(xiàn)已廣泛運(yùn)用于各行各業(yè)的客戶(hù),其中包括:發(fā)電機(jī)回收等企業(yè),備受客戶(hù)贊揚(yáng)。主流應(yīng)用架構(gòu)
客戶(hù)端在對(duì)數(shù)據(jù)庫(kù)發(fā)起請(qǐng)求時(shí),先到緩存層查看是否有所需的數(shù)據(jù),如果緩存層存有客戶(hù)端所需的數(shù)據(jù),則直接從緩存層返回,否則進(jìn)行穿透查詢(xún),對(duì)數(shù)據(jù)庫(kù)進(jìn)行查詢(xún),如果在數(shù)據(jù)庫(kù)中查詢(xún)到該數(shù)據(jù),則將該數(shù)據(jù)回寫(xiě)到緩存層,以便下次客戶(hù)端再次查詢(xún)能夠直接從緩存層獲取數(shù)據(jù)。
緩存中間件 -- Memcache和Redis的區(qū)別
Memcache:代碼層類(lèi)似Hash
1.支持簡(jiǎn)單數(shù)據(jù)類(lèi)型
2.不支持?jǐn)?shù)據(jù)持久化存儲(chǔ)
3.不支持主從
4.不支持分片
Redis
1.數(shù)據(jù)類(lèi)型豐富
2.支持?jǐn)?shù)據(jù)磁盤(pán)持久化存儲(chǔ)
3.支持主從
4.支持分片
為什么Redis能這么快
Redis的效率很高,官方給出的數(shù)據(jù)是100000+QPS(query per second),這是因?yàn)椋?/p>
1.Redis完全基于內(nèi)存,絕大部分請(qǐng)求是純粹的內(nèi)存操作,執(zhí)行效率高。
2.Redis使用單進(jìn)程單線(xiàn)程模型的(K,V)數(shù)據(jù)庫(kù),將數(shù)據(jù)存儲(chǔ)在內(nèi)存中,存取均不會(huì)受到硬盤(pán)IO的限制,因此其執(zhí)行速度極快,另外單線(xiàn)程也能處理高并發(fā)請(qǐng)求,還可以避免頻繁上下文切換和鎖的競(jìng)爭(zhēng),如果想要多核運(yùn)行也可以啟動(dòng)多個(gè)實(shí)例。
3.數(shù)據(jù)結(jié)構(gòu)簡(jiǎn)單,對(duì)數(shù)據(jù)操作也簡(jiǎn)單,Redis不使用表,不會(huì)強(qiáng)制用戶(hù)對(duì)各個(gè)關(guān)系進(jìn)行關(guān)聯(lián),不會(huì)有復(fù)雜的關(guān)系限制,其存儲(chǔ)結(jié)構(gòu)就是鍵值對(duì),類(lèi)似于HashMap,HashMap大的優(yōu)點(diǎn)就是存取的時(shí)間復(fù)雜度為O(1)。
4.Redis使用多路I/O復(fù)用模型,為非阻塞IO(非阻塞IO會(huì)另寫(xiě)一篇解釋?zhuān)梢韵刃邪俣龋?/p>
注:Redis采用的I/O多路復(fù)用函數(shù):epoll/kqueue/evport/select
選用策略:
1.因地制宜,優(yōu)先選擇時(shí)間復(fù)雜度為O(1)的I/O多路復(fù)用函數(shù)作為底層實(shí)現(xiàn)。
2.由于select要遍歷每一個(gè)IO,所以其時(shí)間復(fù)雜度為O(n),通常被作為保底方案。
3.基于react設(shè)計(jì)模式監(jiān)聽(tīng)I(yíng)/O事件。
String
最基本的數(shù)據(jù)類(lèi)型,其值大可存儲(chǔ)512M,二進(jìn)制安全(Redis的String可以包含任何二進(jìn)制數(shù)據(jù),包含jpg對(duì)象等)。
注:如果重復(fù)寫(xiě)入key相同的鍵值對(duì),后寫(xiě)入的會(huì)將之前寫(xiě)入的覆蓋。
Hash
String元素組成的字典,適用于存儲(chǔ)對(duì)象。
List
列表,按照String元素插入順序排序。其順序?yàn)楹筮M(jìn)先出。由于其具有棧的特性,所以可以實(shí)現(xiàn)如“最新消息排行榜”這類(lèi)的功能。
Set
String元素組成的無(wú)序集合,通過(guò)哈希表實(shí)現(xiàn)(增刪改查時(shí)間復(fù)雜度為O(1)),不允許重復(fù)。
另外,當(dāng)我們使用smembers遍歷set中的元素時(shí),其順序也是不確定的,是通過(guò)hash運(yùn)算過(guò)后的結(jié)果。Redis還對(duì)集合提供了求交集、并集、差集等操作,可以實(shí)現(xiàn)如同共同關(guān)注,共同好友等功能。
Sorted Set
通過(guò)分?jǐn)?shù)來(lái)為集合中的成員進(jìn)行從小到大的排序。
更高級(jí)的Redis類(lèi)型
用于計(jì)數(shù)的HyperLogLog、用于支持存儲(chǔ)地理位置信息的Geo。
從海量Key里查詢(xún)出某一個(gè)固定前綴的Key
假設(shè)redis中有十億條key,如何從這么多key中找到固定前綴的key?
方法1:使用KEYS [pattern]:查找所有符合給定模式pattern的key
使用keys [pattern]指令可以找到所有符合pattern條件的key,但是keys會(huì)一次性返回所有符合條件的key,所以會(huì)造成redis的卡頓,假設(shè)redis此時(shí)正在生產(chǎn)環(huán)境下,使用該命令就會(huì)造成隱患,另外如果一次性返回所有key,對(duì)內(nèi)存的消耗在某些條件下也是巨大的。 例:
keys test* //返回所有以test為前綴的key
方法2:使用SCAN cursor [MATCH pattern] [COUNT count]
cursor:游標(biāo) MATCH pattern:查詢(xún)key的條件 count:返回的條數(shù) SCAN是一個(gè)基于游標(biāo)的迭代器,需要基于上一次的游標(biāo)延續(xù)之前的迭代過(guò)程。SCAN以0作為游標(biāo),開(kāi)始一次新的迭代,直到命令返回游標(biāo)0完成一次遍歷。此命令并不保證每次執(zhí)行都返回某個(gè)給定數(shù)量的元素,甚至?xí)祷?個(gè)元素,但只要游標(biāo)不是0,程序都不會(huì)認(rèn)為SCAN命令結(jié)束,但是返回的元素?cái)?shù)量大概率符合count參數(shù)。另外,SCAN支持模糊查詢(xún)。 例:
SCAN 0 MATCH test* COUNT 10 //每次返回10條以test為前綴的key
如何通過(guò)Redis實(shí)現(xiàn)分布式鎖
分布式鎖
分布式鎖是控制分布式系統(tǒng)之間共同訪(fǎng)問(wèn)共享資源的一種鎖的實(shí)現(xiàn)。如果一個(gè)系統(tǒng),或者不同系統(tǒng)的不同主機(jī)之間共享某個(gè)資源時(shí),往往需要互斥,來(lái)排除干擾,滿(mǎn)足數(shù)據(jù)一致性。
分布式鎖需要解決的問(wèn)題如下:
1.互斥性:任意時(shí)刻只有一個(gè)客戶(hù)端獲取到鎖,不能有兩個(gè)客戶(hù)端同時(shí)獲取到鎖。
2.安全性:鎖只能被持有該鎖的客戶(hù)端刪除,不能由其它客戶(hù)端刪除。
3.死鎖:獲取鎖的客戶(hù)端因?yàn)槟承┰蚨礄C(jī)繼而無(wú)法釋放鎖,其它客戶(hù)端再也無(wú)法獲取鎖而導(dǎo)致死鎖,此時(shí)需要有特殊機(jī)制來(lái)避免死鎖。
4.容錯(cuò):當(dāng)各個(gè)節(jié)點(diǎn),如某個(gè)redis節(jié)點(diǎn)宕機(jī)的時(shí)候,客戶(hù)端仍然能夠獲取鎖或釋放鎖。
如何使用redis實(shí)現(xiàn)分布式鎖
使用SETNX實(shí)現(xiàn)
SETNX key value:如果key不存在,則創(chuàng)建并賦值。該命令時(shí)間復(fù)雜度為O(1),如果設(shè)置成功,則返回1,否則返回0。
由于SETNX指令操作簡(jiǎn)單,且是原子性的,所以初期的時(shí)候經(jīng)常被人們作為分布式鎖,我們?cè)趹?yīng)用的時(shí)候,可以在某個(gè)共享資源區(qū)之前先使用SETNX指令,查看是否設(shè)置成功,如果設(shè)置成功則說(shuō)明前方?jīng)]有客戶(hù)端正在訪(fǎng)問(wèn)該資源,如果設(shè)置失敗則說(shuō)明有客戶(hù)端正在訪(fǎng)問(wèn)該資源,那么當(dāng)前客戶(hù)端就需要等待。但是如果真的這么做,就會(huì)存在一個(gè)問(wèn)題,因?yàn)镾ETNX是長(zhǎng)久存在的,所以假設(shè)一個(gè)客戶(hù)端正在訪(fǎng)問(wèn)資源,并且上鎖,那么當(dāng)這個(gè)客戶(hù)端結(jié)束訪(fǎng)問(wèn)時(shí),該鎖依舊存在,后來(lái)者也無(wú)法成功獲取鎖,這個(gè)該如何解決呢?
由于SETNX并不支持傳入EXPIRE參數(shù),所以我們可以直接使用EXPIRE指令來(lái)對(duì)特定的key來(lái)設(shè)置過(guò)期時(shí)間。
用法:EXPIRE key seconds
程序:
RedisService?redisService?=?SpringUtils.getBean(RedisService.class); long?status?=?redisService.setnx(key,"1"); if(status?==?1){ ?redisService.expire(key,expire); ?doOcuppiedWork(); }
這段程序存在的問(wèn)題:假設(shè)程序運(yùn)行到第二行出現(xiàn)異常,那么程序來(lái)不及設(shè)置過(guò)期時(shí)間就結(jié)束了,則key會(huì)一直存在,等同于鎖一直被持有無(wú)法釋放。出現(xiàn)此問(wèn)題的根本原因?yàn)椋涸有缘貌坏綕M(mǎn)足。
解決:從Redis2.6.12版本開(kāi)始,我們就可以使用Set操作,將Setnx和expire融合在一起執(zhí)行,具體做法如下。
SET?KEY?value?[EX?seconds]?[PX?milliseconds]?[NX|XX]
EX second:設(shè)置鍵的過(guò)期時(shí)間為second秒。
PX millisecond:設(shè)置鍵的過(guò)期時(shí)間為millisecond毫秒。
NX:只在鍵不存在時(shí),才對(duì)鍵進(jìn)行設(shè)置操作。
XX:只在鍵已經(jīng)存在時(shí),才對(duì)鍵進(jìn)行設(shè)置操作。
注:SET操作成功完成時(shí)才會(huì)返回OK,否則返回nil。
有了SET我們就可以在程序中使用類(lèi)似下面的代碼實(shí)現(xiàn)分布式鎖了:
RedisService?redisService?=?SpringUtils.getBean(RedisService.class); String?result?=?redisService.set(lockKey,requestId,SET_IF_NOT_EXIST,SET_WITH_EXPIRE_TIME,expireTime); if("OK.equals(result)"){ ?doOcuppiredWork(); }
如何實(shí)現(xiàn)異步隊(duì)列
使用Redis中的List作為隊(duì)列
使用上文所說(shuō)的Redis的數(shù)據(jù)結(jié)構(gòu)中的List作為隊(duì)列 Rpush生產(chǎn)消息,LPOP消費(fèi)消息。
此時(shí)我們可以看到,該隊(duì)列是使用rpush生產(chǎn)隊(duì)列,使用lpop消費(fèi)隊(duì)列。在這個(gè)生產(chǎn)者-消費(fèi)者隊(duì)列里,當(dāng)lpop沒(méi)有消息時(shí),證明該隊(duì)列中沒(méi)有元素,并且生產(chǎn)者還沒(méi)有來(lái)得及生產(chǎn)新的數(shù)據(jù)。
缺點(diǎn):lpop不會(huì)等待隊(duì)列中有值之后再消費(fèi),而是直接進(jìn)行消費(fèi)。
彌補(bǔ):可以通過(guò)在應(yīng)用層引入Sleep機(jī)制去調(diào)用LPOP重試。
使用BLPOP key [key...] timeout
BLPOP key [key ...] timeout:阻塞直到隊(duì)列有消息或者超時(shí)。
缺點(diǎn):按照此種方法,我們生產(chǎn)后的數(shù)據(jù)只能提供給各個(gè)單一消費(fèi)者消費(fèi)能否實(shí)現(xiàn)生產(chǎn)一次就能讓多個(gè)消費(fèi)者消費(fèi)呢?
pub/sub:主題訂閱者模式
發(fā)送者(pub)發(fā)送消息,訂閱者(sub)接收消息。 訂閱者可以訂閱任意數(shù)量的頻道
pub/sub模式的缺點(diǎn):
消息的發(fā)布是無(wú)狀態(tài)的,無(wú)法保證可達(dá)。對(duì)于發(fā)布者來(lái)說(shuō),消息是“即發(fā)即失”的,此時(shí)如果某個(gè)消費(fèi)者在生產(chǎn)者發(fā)布消息時(shí)下線(xiàn),重新上線(xiàn)之后,是無(wú)法接收該消息的,要解決該問(wèn)題需要使用專(zhuān)業(yè)的消息隊(duì)列,如kafka...此處不再贅述。
Redis持久化
什么是持久化
持久化,即將數(shù)據(jù)持久存儲(chǔ),而不因斷電或其它各種復(fù)雜外部環(huán)境影響數(shù)據(jù)的完整性。由于Redis將數(shù)據(jù)存儲(chǔ)在內(nèi)存而不是磁盤(pán)中,所以?xún)?nèi)存一旦斷電,Redis中存儲(chǔ)的數(shù)據(jù)也隨即消失,這往往是用戶(hù)不期望的,所以Redis有持久化機(jī)制來(lái)保證數(shù)據(jù)的安全性。
Redis如何做持久化
Redis目前有兩種持久化方式,即RDB和AOF,RDB是通過(guò)保存某個(gè)時(shí)間點(diǎn)的全量數(shù)據(jù)快照實(shí)現(xiàn)數(shù)據(jù)的持久化,當(dāng)恢復(fù)數(shù)據(jù)時(shí),直接通過(guò)rdb文件中的快照,將數(shù)據(jù)恢復(fù)。
RDB(快照)持久化:保存某個(gè)時(shí)間點(diǎn)的全量數(shù)據(jù)快照
RDB持久化會(huì)在某個(gè)特定的間隔保存那個(gè)時(shí)間點(diǎn)的全量數(shù)據(jù)的快照。 RDB配置文件: redis.conf:
save?900?1?#在900s內(nèi)如果有1條數(shù)據(jù)被寫(xiě)入,則產(chǎn)生一次快照。 ?save?300?10?#在300s內(nèi)如果有10條數(shù)據(jù)被寫(xiě)入,則產(chǎn)生一次快照 ?save?60?10000?#在60s內(nèi)如果有10000條數(shù)據(jù)被寫(xiě)入,則產(chǎn)生一次快照 ?stop-writes-on-bgsave-error?yes? ?#stop-writes-on-bgsave-error?: ?如果為yes則表示,當(dāng)備份進(jìn)程出錯(cuò)的時(shí)候, ?主進(jìn)程就停止進(jìn)行接受新的寫(xiě)入操作,這樣是為了保護(hù)持久化的數(shù)據(jù)一致性的問(wèn)題。
RDB的創(chuàng)建與載入
SAVE:阻塞Redis的服務(wù)器進(jìn)程,直到RDB文件被創(chuàng)建完畢。SAVE命令很少被使用,因?yàn)槠鋾?huì)阻塞主線(xiàn)程來(lái)保證快照的寫(xiě)入,由于Redis是使用一個(gè)主線(xiàn)程來(lái)接收所有客戶(hù)端請(qǐng)求,這樣會(huì)阻塞所有客戶(hù)端請(qǐng)求。
BGSAVE:該指令會(huì)Fork出一個(gè)子進(jìn)程來(lái)創(chuàng)建RDB文件,不阻塞服務(wù)器進(jìn)程,子進(jìn)程接收請(qǐng)求并創(chuàng)建RDB快照,父進(jìn)程繼續(xù)接收客戶(hù)端的請(qǐng)求。子進(jìn)程在完成文件的創(chuàng)建時(shí)會(huì)向父進(jìn)程發(fā)送信號(hào),父進(jìn)程在接收客戶(hù)端請(qǐng)求的過(guò)程中,在一定的時(shí)間間隔通過(guò)輪詢(xún)來(lái)接收子進(jìn)程的信號(hào)。我們也可以通過(guò)使用lastsave指令來(lái)查看bgsave是否執(zhí)行成功,lastsave可以返回最后一次執(zhí)行成功bgsave的時(shí)間。
自動(dòng)化觸發(fā)RDB持久化的方式
1.根據(jù)redis.conf配置里的SAVE m n 定時(shí)觸發(fā)(實(shí)際上使用的是BGSAVE)
2.主從復(fù)制時(shí),主節(jié)點(diǎn)自動(dòng)觸發(fā)。
3.執(zhí)行Debug Reload
4.執(zhí)行Shutdown且沒(méi)有開(kāi)啟AOF持久化。
BGSAVE的原理
啟動(dòng):
1.檢查是否存在子進(jìn)程正在執(zhí)行AOF或者RDB的持久化任務(wù)。如果有則返回false。
2.調(diào)用Redis源碼中的rdbSaveBackground方法,方法中執(zhí)行fork()產(chǎn)生子進(jìn)程執(zhí)行rdb操作。
3.關(guān)于fork()中的Copy-On-Write
fork()在linux中創(chuàng)建子進(jìn)程采用Copy-On-Write(寫(xiě)時(shí)拷貝技術(shù)),即如果有多個(gè)調(diào)用者同時(shí)要求相同資源(如內(nèi)存或磁盤(pán)上的數(shù)據(jù)存儲(chǔ)),他們會(huì)共同獲取相同的指針指向相同的資源,直到某個(gè)調(diào)用者試圖修改資源的內(nèi)容時(shí),系統(tǒng)才會(huì)真正復(fù)制一份專(zhuān)用副本給調(diào)用者,而其它調(diào)用者所見(jiàn)到的最初的資源仍然保持不變。
RDB持久化方式的缺點(diǎn)
1.內(nèi)存數(shù)據(jù)全量同步,數(shù)據(jù)量大的狀況下,會(huì)由于I/O而嚴(yán)重影響性能。
2.可能會(huì)因?yàn)镽edis宕機(jī)而丟失從當(dāng)前至最近一次快照期間的數(shù)據(jù)。
AOF(Append-Only-File)持久化:保存寫(xiě)狀態(tài)
AOF持久化是通過(guò)保存Redis的寫(xiě)狀態(tài)來(lái)記錄數(shù)據(jù)庫(kù)的。相對(duì)RDB來(lái)說(shuō),RDB持久化是通過(guò)備份數(shù)據(jù)庫(kù)的狀態(tài)來(lái)記錄數(shù)據(jù)庫(kù),而AOF持久化是備份數(shù)據(jù)庫(kù)接收到的指令。
1.AOF記錄除了查詢(xún)以外的所有變更數(shù)據(jù)庫(kù)狀態(tài)的指令。
2.以增量的形式追加保存到AOF文件中。
開(kāi)啟AOF持久化
1.打開(kāi)redis.conf配置文件,將appendonly屬性改為yes。
2.修改appendfsync屬性,該屬性可以接收三種參數(shù),分別是always,everysec,no,always表示總是即時(shí)將緩沖區(qū)內(nèi)容寫(xiě)入AOF文件當(dāng)中,everysec表示每隔一秒將緩沖區(qū)內(nèi)容寫(xiě)入AOF文件,no表示將寫(xiě)入文件操作交由操作系統(tǒng)決定,一般來(lái)說(shuō),操作系統(tǒng)考慮效率問(wèn)題,會(huì)等待緩沖區(qū)被填滿(mǎn)再將緩沖區(qū)數(shù)據(jù)寫(xiě)入AOF文件中。
appendonly?yes ?#appendsync?always ?appendfsync?everysec ?#?appendfsync?no
日志重寫(xiě)解決AOF文件不斷增大的問(wèn)題
隨著寫(xiě)操作的不斷增加,AOF文件會(huì)越來(lái)越大。假設(shè)遞增一個(gè)計(jì)數(shù)器100次,如果使用RDB持久化方式,我們只要保存最終結(jié)果100即可,而AOF持久化方式需要記錄下這100次遞增操作的指令,而事實(shí)上要恢復(fù)這條記錄,只需要執(zhí)行一條命令就行,所以那一百條命令實(shí)際可以精簡(jiǎn)為一條。Redis支持這樣的功能,在不中斷前臺(tái)服務(wù)的情況下,可以重寫(xiě)AOF文件,同樣使用到了COW(寫(xiě)時(shí)拷貝)。重寫(xiě)過(guò)程如下:
1.調(diào)用fork(),創(chuàng)建一個(gè)子進(jìn)程。
2.子進(jìn)程把新的AOF寫(xiě)到一個(gè)臨時(shí)文件里,不依賴(lài)原來(lái)的AOF文件。
3.主進(jìn)程持續(xù)將新的變動(dòng)同時(shí)寫(xiě)到內(nèi)存和原來(lái)的AOF里。
4.主進(jìn)程獲取子進(jìn)程重寫(xiě)AOF的完成信號(hào),往新AOF同步增量變動(dòng)。
5.使用新的AOF文件替換掉舊的AOF文件。
AOF和RDB的優(yōu)缺點(diǎn)
RDB優(yōu)點(diǎn):全量數(shù)據(jù)快照,文件小,恢復(fù)快。
RDB缺點(diǎn):無(wú)法保存最近一次快照之后的數(shù)據(jù)。
AOF優(yōu)點(diǎn):可讀性高,適合保存增量數(shù)據(jù),數(shù)據(jù)不易丟失。
AOF缺點(diǎn):文件體積大,恢復(fù)時(shí)間長(zhǎng)。
RDB-AOF混合持久化方式
redis4.0之后推出了此種持久化方式,RDB作為全量備份,AOF作為增量備份,并且將此種方式作為默認(rèn)方式使用。
在上述兩種方式中,RDB方式是將全量數(shù)據(jù)寫(xiě)入RDB文件,這樣寫(xiě)入的特點(diǎn)是文件小,恢復(fù)快,但無(wú)法保存最近一次快照之后的數(shù)據(jù),AOF則將redis指令存入文件中,這樣又會(huì)造成文件體積大,恢復(fù)時(shí)間長(zhǎng)等弱點(diǎn)。
在RDB-AOF方式下,持久化策略首先將緩存中數(shù)據(jù)以RDB方式全量寫(xiě)入文件,再將寫(xiě)入后新增的數(shù)據(jù)以AOF的方式追加在RDB數(shù)據(jù)的后面,在下一次做RDB持久化的時(shí)候?qū)OF的數(shù)據(jù)重新以RDB的形式寫(xiě)入文件。這種方式既可以提高讀寫(xiě)和恢復(fù)效率,也可以減少文件大小,同時(shí)可以保證數(shù)據(jù)的完整性。在此種策略的持久化過(guò)程中,子進(jìn)程會(huì)通過(guò)管道從父進(jìn)程讀取增量數(shù)據(jù),在以RDB格式保存全量數(shù)據(jù)時(shí),也會(huì)通過(guò)管道讀取數(shù)據(jù),同時(shí)不會(huì)造成管道阻塞??梢哉f(shuō),在此種方式下的持久化文件,前半段是RDB格式的全量數(shù)據(jù),后半段是AOF格式的增量數(shù)據(jù)。此種方式是目前較為推薦的一種持久化方式。
Redis數(shù)據(jù)的恢復(fù)
RDB和AOF文件共存情況下的恢復(fù)流程
從圖可知,Redis啟動(dòng)時(shí)會(huì)先檢查AOF是否存在,如果AOF存在則直接加載AOF,如果不存在A(yíng)OF,則直接加載RDB文件。
Pineline
Pipeline和Linux的管道類(lèi)似,它可以讓Redis批量執(zhí)行指令。
Redis基于請(qǐng)求/響應(yīng)模型,單個(gè)請(qǐng)求處理需要一一應(yīng)答。如果需要同時(shí)執(zhí)行大量命令,則每條命令都需要等待上一條命令執(zhí)行完畢后才能繼續(xù)執(zhí)行,這中間不僅僅多了RTT,還多次使用了系統(tǒng)IO。Pipeline由于可以批量執(zhí)行指令,所以可以節(jié)省多次IO和請(qǐng)求響應(yīng)往返的時(shí)間。但是如果指令之間存在依賴(lài)關(guān)系,則建議分批發(fā)送指令。
Redis的同步機(jī)制
主從同步原理
Redis一般是使用一個(gè)Master節(jié)點(diǎn)來(lái)進(jìn)行寫(xiě)操作,而若干個(gè)Slave節(jié)點(diǎn)進(jìn)行讀操作,Master和Slave分別代表了一個(gè)個(gè)不同的RedisServer實(shí)例,另外定期的數(shù)據(jù)備份操作也是單獨(dú)選擇一個(gè)Slave去完成,這樣可以大程度發(fā)揮Redis的性能,為的是保證數(shù)據(jù)的弱一致性和最終一致性。另外,Master和Slave的數(shù)據(jù)不是一定要即時(shí)同步的,但是在一段時(shí)間后Master和Slave的數(shù)據(jù)是趨于同步的,這就是最終一致性。
全同步過(guò)程
1.Slave發(fā)送sync命令到Master。
2.Master啟動(dòng)一個(gè)后臺(tái)進(jìn)程,將Redis中的數(shù)據(jù)快照保存到文件中。
3.Master將保存數(shù)據(jù)快照期間接收到的寫(xiě)命令緩存起來(lái)。
4.Master完成寫(xiě)文件操作后,將該文件發(fā)送給Slave。
5.使用新的AOF文件替換掉舊的AOF文件。
6.Master將這期間收集的增量寫(xiě)命令發(fā)送給Slave端。
增量同步過(guò)程
1.Master接收到用戶(hù)的操作指令,判斷是否需要傳播到Slave。
2.將操作記錄追加到AOF文件。
3.將操作傳播到其它Slave:1.對(duì)齊主從庫(kù);2.往響應(yīng)緩存寫(xiě)入指令。
4.將緩存中的數(shù)據(jù)發(fā)送給Slave。
Redis Sentinel(哨兵)
主從模式弊端:當(dāng)Master宕機(jī)后,Redis集群將不能對(duì)外提供寫(xiě)入操作。Redis Sentinel可解決這一問(wèn)題。
解決主從同步Master宕機(jī)后的主從切換問(wèn)題:
1.監(jiān)控:檢查主從服務(wù)器是否運(yùn)行正常。
2.提醒:通過(guò)API向管理員或者其它應(yīng)用程序發(fā)送故障通知。
3.自動(dòng)故障遷移:主從切換(在Master宕機(jī)后,將其中一個(gè)Slave轉(zhuǎn)為Master,其他的
Slave從該節(jié)點(diǎn)同步數(shù)據(jù))。
Redis集群
原理:如何從海量數(shù)據(jù)里快速找到所需?
分片
按照某種規(guī)則去劃分?jǐn)?shù)據(jù),分散存儲(chǔ)在多個(gè)節(jié)點(diǎn)上。通過(guò)將數(shù)據(jù)分到多個(gè)Redis服務(wù)器上,來(lái)減輕單個(gè)Redis服務(wù)器的壓力。
一致性Hash算法
既然要將數(shù)據(jù)進(jìn)行分片,那么通常的做法就是獲取節(jié)點(diǎn)的Hash值,然后根據(jù)節(jié)點(diǎn)數(shù)求模,但這樣的方法有明顯的弊端,當(dāng)Redis節(jié)點(diǎn)數(shù)需要?jiǎng)討B(tài)增加或減少的時(shí)候,會(huì)造成大量的Key無(wú)法被命中。所以Redis中引入了一致性Hash算法。該算法對(duì)2^32 取模,將Hash值空間組成虛擬的圓環(huán),整個(gè)圓環(huán)按順時(shí)針?lè)较蚪M織,每個(gè)節(jié)點(diǎn)依次為0、1、2...2^32-1,之后將每個(gè)服務(wù)器進(jìn)行Hash運(yùn)算,確定服務(wù)器在這個(gè)Hash環(huán)上的地址,確定了服務(wù)器地址后,對(duì)數(shù)據(jù)使用同樣的Hash算法,將數(shù)據(jù)定位到特定的Redis服務(wù)器上。如果定位到的地方?jīng)]有Redis服務(wù)器實(shí)例,則繼續(xù)順時(shí)針尋找,找到的第一臺(tái)服務(wù)器即該數(shù)據(jù)最終的服務(wù)器位置。
Hash環(huán)的數(shù)據(jù)傾斜問(wèn)題
Hash環(huán)在服務(wù)器節(jié)點(diǎn)很少的時(shí)候,容易遇到服務(wù)器節(jié)點(diǎn)不均勻的問(wèn)題,這會(huì)造成數(shù)據(jù)傾斜,數(shù)據(jù)傾斜指的是被緩存的對(duì)象大部分集中在Redis集群的其中一臺(tái)或幾臺(tái)服務(wù)器上。
如上圖,一致性Hash算法運(yùn)算后的數(shù)據(jù)大部分被存放在A(yíng)節(jié)點(diǎn)上,而B(niǎo)節(jié)點(diǎn)只存放了少量的數(shù)據(jù),久而久之A節(jié)點(diǎn)將被撐爆。
針對(duì)這一問(wèn)題,可以引入虛擬節(jié)點(diǎn)解決。簡(jiǎn)單地說(shuō),就是為每一個(gè)服務(wù)器節(jié)點(diǎn)計(jì)算多個(gè)Hash,每個(gè)計(jì)算結(jié)果位置都放置一個(gè)此服務(wù)器節(jié)點(diǎn),稱(chēng)為虛擬節(jié)點(diǎn),可以在服務(wù)器IP或者主機(jī)名后放置一個(gè)編號(hào)實(shí)現(xiàn)。
例如上圖:將NodeA和NodeB兩個(gè)節(jié)點(diǎn)分為Node A#1-A#3 NodeB#1-B#3。
結(jié)語(yǔ)
這篇準(zhǔn)(tou)備(lan)了相當(dāng)久的時(shí)間,因?yàn)橛行〇|西總感覺(jué)自己拿不準(zhǔn)不敢往上寫(xiě),差點(diǎn)自閉,就算現(xiàn)在發(fā)出來(lái)了也感覺(jué)有很多地方是需要改動(dòng)的。如果有同學(xué)覺(jué)得哪里寫(xiě)的不對(duì)勁的,評(píng)論區(qū)或者私聊我...嗯,我不要你覺(jué)得,我要我覺(jué)得。
創(chuàng)新互聯(lián)www.cdcxhl.cn,專(zhuān)業(yè)提供香港、美國(guó)云服務(wù)器,動(dòng)態(tài)BGP最優(yōu)骨干路由自動(dòng)選擇,持續(xù)穩(wěn)定高效的網(wǎng)絡(luò)助力業(yè)務(wù)部署。公司持有工信部辦法的idc、isp許可證, 機(jī)房獨(dú)有T級(jí)流量清洗系統(tǒng)配攻擊溯源,準(zhǔn)確進(jìn)行流量調(diào)度,確保服務(wù)器高可用性。佳節(jié)活動(dòng)現(xiàn)已開(kāi)啟,新人活動(dòng)云服務(wù)器買(mǎi)多久送多久。