這篇文章將為大家詳細(xì)講解有關(guān)關(guān)于redis緩存的面試題有哪些,小編覺(jué)得挺實(shí)用的,因此分享給大家做個(gè)參考,希望大家閱讀完這篇文章后可以有所收獲。
目前成都創(chuàng)新互聯(lián)公司已為成百上千家的企業(yè)提供了網(wǎng)站建設(shè)、域名、網(wǎng)頁(yè)空間、成都網(wǎng)站托管、企業(yè)網(wǎng)站設(shè)計(jì)、鼓樓網(wǎng)站維護(hù)等服務(wù),公司將堅(jiān)持客戶導(dǎo)向、應(yīng)用為本的策略,正道將秉承"和諧、參與、激情"的文化,與客戶和合作伙伴齊心協(xié)力一起成長(zhǎng),共同發(fā)展。
區(qū)別:
memcached可緩存圖片和視頻。redis支持除k/v更多的數(shù)據(jù)結(jié)構(gòu);
redis可以使用虛擬內(nèi)存,redis可持久化和aof災(zāi)難恢復(fù),redis通過(guò)主從支持?jǐn)?shù)據(jù)備份;
3.redis可以做消息隊(duì)列。
原因:memcached多線程模型引入了緩存一致性和鎖,加鎖帶來(lái)了性能損耗。
主從復(fù)制實(shí)現(xiàn):主節(jié)點(diǎn)將自己內(nèi)存中的數(shù)據(jù)做一份快照,將快照發(fā)給從節(jié)點(diǎn),從節(jié)點(diǎn)將數(shù)據(jù)恢復(fù)到內(nèi)存中。之后再每次增加新數(shù)據(jù)的時(shí)候,主節(jié)點(diǎn)以類似于MySQL的二進(jìn)制日志方式將語(yǔ)句發(fā)送給從節(jié)點(diǎn),從節(jié)點(diǎn)拿到主節(jié)點(diǎn)發(fā)送過(guò)來(lái)的語(yǔ)句進(jìn)行重放。
分片方式:
客戶端分片
基于代理的分片
Twemproxy
codis
路由查詢分片
Redis-cluster體身提供了自動(dòng)將數(shù)據(jù)分散到RedisCluster不同節(jié)點(diǎn)的能力,整個(gè)數(shù)據(jù)集合的某個(gè)數(shù)據(jù)子集存儲(chǔ)在哪個(gè)節(jié)點(diǎn)對(duì)于用戶來(lái)說(shuō)是透明的)
redis-cluster分片原理:Cluster中有一個(gè)16384長(zhǎng)度的槽(虛擬槽),編號(hào)分別為0-16383。每個(gè)Master節(jié)點(diǎn)都會(huì)負(fù)責(zé)一部分的槽,當(dāng)有某個(gè)key被映射到某個(gè)Master負(fù)責(zé)的槽,那么這個(gè)Master負(fù)責(zé)為這個(gè)key提供服務(wù),至于哪個(gè)Master節(jié)點(diǎn)負(fù)責(zé)哪個(gè)槽,可以由用戶指定,也可以在初始化的時(shí)候自動(dòng)生成,只有Master才擁有槽的所有權(quán)。Master節(jié)點(diǎn)維護(hù)著一個(gè)16384/8字節(jié)的位序列,Master節(jié)點(diǎn)用bit來(lái)標(biāo)識(shí)對(duì)于某個(gè)槽自己是否擁有。比如對(duì)于編號(hào)為1的槽,Master只要判斷序列的第二位(索引從0開(kāi)始)是不是為1即可。這種結(jié)構(gòu)很容易添加或者刪除節(jié)點(diǎn)。比如如果我想新添加個(gè)節(jié)點(diǎn)D,我需要從節(jié)點(diǎn)A、B、C中得部分槽到D上。
【相關(guān)推薦:Redis視頻教程】
redis:
線程Asetnx(上鎖的對(duì)象超時(shí)時(shí)的時(shí)間戳tl),如果返回true,獲得鎖。
線程B用get獲取t1,與當(dāng)前時(shí)間戳比較,判斷是是否超時(shí),沒(méi)超時(shí)false,若超時(shí)執(zhí)行第3步;
計(jì)算新的超時(shí)時(shí)間t2,使用getset命令返回t3(該值可能其他線程已經(jīng)修改過(guò)),如果t1==t3,獲得鎖,如果t1!=t3說(shuō)明鎖被其他線程獲取了。
獲取鎖后,處理完業(yè)務(wù)邏輯,再去判斷鎖是否超時(shí),如果沒(méi)超時(shí)刪除鎖,如果已超時(shí),不用處理(防止刪除其他線程的鎖)。
zk:
客戶端對(duì)某個(gè)方法加鎖時(shí),在zk上的與該方法對(duì)應(yīng)的指定節(jié)點(diǎn)的目錄下,生成一個(gè)唯一的瞬時(shí)有序節(jié)點(diǎn)node1;
客戶端獲取該路徑下所有已經(jīng)創(chuàng)建的子節(jié)點(diǎn),如果發(fā)現(xiàn)自己創(chuàng)建的node1的序號(hào)是最小的,就認(rèn)為這個(gè)客戶端獲得了鎖。
如果發(fā)現(xiàn)node1不是最小的,則監(jiān)聽(tīng)比自己創(chuàng)建節(jié)點(diǎn)序號(hào)小的最大的節(jié)點(diǎn),進(jìn)入等待。
獲取鎖后,處理完邏輯,刪除自己創(chuàng)建的node1即可。區(qū)別:zk性能差一些,開(kāi)銷大,實(shí)現(xiàn)簡(jiǎn)單。
RDB(RedisDataBase:在不同的時(shí)間點(diǎn)將redis的數(shù)據(jù)生成的快照同步到磁盤等介質(zhì)上):內(nèi)存到硬盤的快照,定期更新。缺點(diǎn):耗時(shí),耗性能(fork+io操作),易丟失數(shù)據(jù)。
AOF(AppendOnlyFile:將redis所執(zhí)行過(guò)的所有指令都記錄下來(lái),在下次redis重啟時(shí),只需要執(zhí)行指令就可以了):寫(xiě)日志。缺點(diǎn):體積大,恢復(fù)速度慢。
bgsave做鏡像全量持久化,aof做增量持久化。因?yàn)閎gsave會(huì)消耗比較長(zhǎng)的時(shí)間,不夠?qū)崟r(shí),在停機(jī)的時(shí)候會(huì)導(dǎo)致大量的數(shù)據(jù)丟失,需要aof來(lái)配合,在redis實(shí)例重啟時(shí),優(yōu)先使用aof來(lái)恢復(fù)內(nèi)存的狀態(tài),如果沒(méi)有aof日志,就會(huì)使用rdb文件來(lái)恢復(fù)。Redis會(huì)定期做aof重寫(xiě),壓縮aof文件日志大小。Redis4.0之后有了混合持久化的功能,將bgsave的全量和aof的增量做了融合處理,這樣既保證了恢復(fù)的效率又兼顧了數(shù)據(jù)的安全性。bgsave的原理,fork和cow,fork是指redis通過(guò)創(chuàng)建子進(jìn)程來(lái)進(jìn)行bgsave操作,cow指的是copyonwrite,子進(jìn)程創(chuàng)建后,父子進(jìn)程共享數(shù)據(jù)段,父進(jìn)程繼續(xù)提供讀寫(xiě)服務(wù),寫(xiě)臟的頁(yè)面數(shù)據(jù)會(huì)逐漸和子進(jìn)程分囲開(kāi)來(lái)。
過(guò)期策略:
定時(shí)過(guò)期(一key一定時(shí)器),惰性過(guò)期:只有使用key時(shí)才判斷key是否已過(guò)期,過(guò)期則清除。定期過(guò)期:前兩者折中。
LRU:newLinkedHashMap
LRU算法實(shí)現(xiàn):
通過(guò)雙向鏈表來(lái)實(shí)現(xiàn),新數(shù)據(jù)插入到鏈表頭部;
每當(dāng)緩存命中(即緩存數(shù)據(jù)被訪問(wèn)),則將數(shù)據(jù)移到鏈表頭部;
當(dāng)鏈表滿的時(shí)候,將鏈表尾部的數(shù)據(jù)丟棄。
LinkedHashMap:HashMap和雙向鏈表合二為一即是LinkedHashMap。HashMap是無(wú)序的,LinkedHashMap通過(guò)維護(hù)一個(gè)額外的雙向鏈表保證了迭代順序。該迭代順序可以是插入順序(默認(rèn)),也可以是訪問(wèn)順序。
**緩存穿透:**指查詢一個(gè)一定不存在的數(shù)據(jù),如果從存儲(chǔ)層查不到數(shù)據(jù)則不寫(xiě)入緩存,這將導(dǎo)致這個(gè)不存在的數(shù)據(jù)每次請(qǐng)求都要到DB去查詢,可能導(dǎo)致DB掛掉。
解決方案:
查詢返回的數(shù)據(jù)為空,仍把這個(gè)空結(jié)果進(jìn)行緩存,但過(guò)期時(shí)間會(huì)比較短;
布隆過(guò)濾器:將所有可能存在的數(shù)據(jù)哈希到一個(gè)足夠大的bitmap中,一個(gè)一定不存在的數(shù)據(jù)會(huì)被這個(gè)bitmap攔截掉,從而避免了對(duì)DB的查詢。
**緩存擊穿:**對(duì)于設(shè)置了過(guò)期時(shí)間的key,緩存在某個(gè)時(shí)間點(diǎn)過(guò)期的時(shí)候,恰好這時(shí)間點(diǎn)對(duì)這個(gè)Key有大量的并發(fā)請(qǐng)求過(guò)來(lái),這些請(qǐng)求發(fā)現(xiàn)緩存過(guò)期一般都會(huì)從后端DB加載數(shù)據(jù)并回設(shè)到緩存,這個(gè)時(shí)候大并發(fā)的請(qǐng)求可能會(huì)瞬間把DB壓垮。
解決方案:
使用互斥鎖:當(dāng)緩存失效時(shí),不立即去Ioaddb,先使用如Redis的setnx去設(shè)置一個(gè)互斥鎖,當(dāng)操作成功返回時(shí)再進(jìn)行Ioaddb的操作并回設(shè)緩存,否則重試get緩存的方法。
永遠(yuǎn)不過(guò)期:物理不過(guò)期,但邏輯過(guò)期(后臺(tái)異步線程去刷新)。緩存雪崩:設(shè)置緩存時(shí)采用了相同的過(guò)期時(shí)間,導(dǎo)致緩存在某一時(shí)刻同時(shí)失效,請(qǐng)求全部轉(zhuǎn)發(fā)到DB,DB瞬時(shí)壓力過(guò)重雪崩。與緩存擊穿的區(qū)別:雪崩是很多key,擊穿是某一個(gè)key緩存。
解決方案:
將緩存失效時(shí)間分散開(kāi),比如可以在原有的失效時(shí)間基礎(chǔ)上增加一個(gè)隨機(jī)值,比如1-5分鐘隨機(jī),這樣每一個(gè)緩存的過(guò)期時(shí)間的重復(fù)率就會(huì)降低,就很難引發(fā)集體失效的事件。
選擇redis的情況:
復(fù)雜數(shù)據(jù)結(jié)構(gòu),value的數(shù)據(jù)是哈希,列表,集合,有序集合等這種情況下,會(huì)選擇redis,因?yàn)閙emcache無(wú)法滿足這些數(shù)據(jù)結(jié)構(gòu),最典型的的使用場(chǎng)景是,用戶訂單列表,用戶消息,帖子評(píng)論等。
需要進(jìn)行數(shù)據(jù)的持久化功能,但是注意,不要把redis當(dāng)成數(shù)據(jù)庫(kù)使用,如果redis掛了,內(nèi)存能夠快速恢復(fù)熱數(shù)據(jù),不會(huì)將壓力瞬間壓在數(shù)據(jù)庫(kù)上,沒(méi)有cache預(yù)熱的過(guò)程。對(duì)于只讀和數(shù)據(jù)一致性要求不高的場(chǎng)景可以采用持久化存儲(chǔ)
高可用,redis支持集群,可以實(shí)現(xiàn)主動(dòng)復(fù)制,讀寫(xiě)分離,而對(duì)于memcache如果想要實(shí)現(xiàn)高可用,需要進(jìn)行二次開(kāi)發(fā)。
存儲(chǔ)的內(nèi)容比較大,memcache存儲(chǔ)的value最大為1M。
選擇memcache的場(chǎng)景:
純KV,數(shù)據(jù)量非常大的業(yè)務(wù),使用memcache更合適,原因是:
memcache的內(nèi)存分配采用的是預(yù)分配內(nèi)存池的管理方式,能夠省去內(nèi)存分配的時(shí)間,redis是臨時(shí)申請(qǐng)空間,可能導(dǎo)致碎片化。
虛擬內(nèi)存使用,memcache將所有的數(shù)據(jù)存儲(chǔ)在物理內(nèi)存里,redis有自己的vm機(jī)制,理論上能夠存儲(chǔ)比物理內(nèi)存更多的數(shù)據(jù),當(dāng)數(shù)據(jù)超量時(shí),引發(fā)swap,把冷數(shù)據(jù)刷新到磁盤上,從這點(diǎn)上,數(shù)據(jù)量大時(shí),memcache更快
網(wǎng)絡(luò)模型,memcache使用非阻塞的10復(fù)用模型,redis也是使用非阻塞的I。復(fù)用模型,但是redis還提供了一些非KV存儲(chǔ)之外的排序,聚合功能,復(fù)雜的CPU計(jì)算,會(huì)阻塞整個(gè)I0調(diào)度,從這點(diǎn)上由于redis提供的功能較多,memcache更快些
線程模型,memcache使用多線程,主線程監(jiān)聽(tīng),worker子線程接受請(qǐng)求,執(zhí)行讀寫(xiě),這個(gè)過(guò)程可能存在鎖沖突。redis使用的單線程,雖然無(wú)鎖沖突,但是難以利用多核的特性提升吞吐量。
假設(shè)采用的主存分離,讀寫(xiě)分離的數(shù)據(jù)庫(kù),
如果一個(gè)線程A先刪除緩存數(shù)據(jù),然后將數(shù)據(jù)寫(xiě)入到主庫(kù)當(dāng)中,這個(gè)時(shí)候,主庫(kù)和從庫(kù)同步?jīng)]有完成,線程B從緩存當(dāng)中讀取數(shù)據(jù)失敗,從從庫(kù)當(dāng)中讀取到舊數(shù)據(jù),然后更新至緩存,這個(gè)時(shí)候,緩存當(dāng)中的就是舊的數(shù)據(jù)。
發(fā)生上述不一致的原因在于,主從庫(kù)數(shù)據(jù)不一致問(wèn)題,加入了緩存之后,主從不一致的時(shí)間被拉長(zhǎng)了。
處理思路:在從庫(kù)有數(shù)據(jù)更新之后,將緩存當(dāng)中的數(shù)據(jù)也同時(shí)進(jìn)行更新,即當(dāng)從庫(kù)發(fā)生了數(shù)據(jù)更新之后,向緩存發(fā)出刪除,淘汰這段時(shí)間寫(xiě)入的舊數(shù)據(jù)。
場(chǎng)景描述,對(duì)于主從庫(kù),讀寫(xiě)分離,如果主從庫(kù)更新同步有時(shí)差,就會(huì)導(dǎo)致主從庫(kù)數(shù)據(jù)的不一致
忽略這個(gè)數(shù)據(jù)不一致,在數(shù)據(jù)一致性要求不高的業(yè)務(wù)下,未必需要時(shí)時(shí)一致性
強(qiáng)制讀主庫(kù),使用一個(gè)高可用的主庫(kù),數(shù)據(jù)庫(kù)讀寫(xiě)都在主庫(kù),添加一個(gè)緩存,提升數(shù)據(jù)讀取的性能。
選擇性讀主庫(kù),添加一個(gè)緩存,用來(lái)記錄必須讀主庫(kù)的數(shù)據(jù),將哪個(gè)庫(kù),哪個(gè)表,哪個(gè)主鍵,作為緩存的key,設(shè)置緩存失效的時(shí)間為主從庫(kù)同步的時(shí)間,如果緩存當(dāng)中有這個(gè)數(shù)據(jù),直接讀取主庫(kù),如果緩存當(dāng)中沒(méi)有這個(gè)主鍵,就到對(duì)應(yīng)的從庫(kù)中讀取。
master最好不要做持久化工作,如RDB內(nèi)存快照和AOF日志文件
如果數(shù)據(jù)比較重要,某個(gè)slave開(kāi)啟AOF備份,策略設(shè)置成每秒同步一次
為了主從復(fù)制的速度和連接的穩(wěn)定性,master和Slave最好在一個(gè)局域網(wǎng)內(nèi)
盡量避免在壓力大得主庫(kù)上增加從庫(kù)
主從復(fù)制不要米用網(wǎng)狀結(jié)構(gòu),盡量是線性結(jié)構(gòu),Master<–Slave1<—Slave2…
voltile-lru從已經(jīng)設(shè)置過(guò)期時(shí)間的數(shù)據(jù)集中挑選最近最少使用的數(shù)據(jù)淘汰
voltile-ttl從已經(jīng)設(shè)置過(guò)期時(shí)間的數(shù)據(jù)庫(kù)集當(dāng)中挑選將要過(guò)期的數(shù)據(jù)
voltile-random從已經(jīng)設(shè)置過(guò)期時(shí)間的數(shù)據(jù)集任意選擇淘汰數(shù)據(jù)
allkeys-lru從數(shù)據(jù)集中挑選最近最少使用的數(shù)據(jù)淘汰
allkeys-random從數(shù)據(jù)集中任意選擇淘汰的數(shù)據(jù)
no-eviction禁止驅(qū)逐數(shù)據(jù)
字符串String、字典Hash、列表List、集合Set、有序集合SortedSet。如果是咼級(jí)用戶,那么還會(huì)有,如果你是Redis中高級(jí)用戶,還需要加上下面幾種數(shù)據(jù)結(jié)構(gòu)HyperLogLog、Geo、Pub/Sub。
使用keys指令可以掃出指定模式的key列表。
對(duì)方接著追問(wèn):如果這個(gè)redis正在給線上的業(yè)務(wù)提供服務(wù),那使用keys指令會(huì)有什么問(wèn)題?
這個(gè)時(shí)候你要回答redis關(guān)鍵的一個(gè)特性:redis的單線程的。keys指令會(huì)導(dǎo)致線程阻塞一段時(shí)間,線上服務(wù)會(huì)停頓,直到指令執(zhí)行完畢,服務(wù)才能恢復(fù)。這個(gè)時(shí)候可以使用scan指令,scan指令可以無(wú)阻塞的提取出指定模式的key列表,但是會(huì)有一定的重復(fù)概率,在客戶端做一次去重就可以了,但是整體所花費(fèi)的時(shí)間會(huì)比直接用keys指令長(zhǎng)。
使用list類型保存數(shù)據(jù)信息,rpush生產(chǎn)消息,lpop消費(fèi)消息,當(dāng)lpop沒(méi)有消息時(shí),可以sleep一段時(shí)間,然后再檢查有沒(méi)有信息,如果不想sleep的話,可以使用blpop,在沒(méi)有信息的時(shí)候,會(huì)一直阻塞,直到信息的到來(lái)。redis可以通過(guò)pub/sub主題訂閱模式實(shí)現(xiàn)一個(gè)生產(chǎn)者,多個(gè)消費(fèi)者,當(dāng)然也存在一定的缺點(diǎn),當(dāng)消費(fèi)者下線時(shí),生產(chǎn)的消息會(huì)丟失。
使用sortedset,使用時(shí)間戳做score,消息內(nèi)容作為key,調(diào)用zadd來(lái)生產(chǎn)消息,消費(fèi)者使用zrangbyscore獲取n秒之前的數(shù)據(jù)做輪詢處理。
Redis本質(zhì)上是一個(gè)Key-Value類型的內(nèi)存數(shù)據(jù)庫(kù),很像memcached,整個(gè)數(shù)據(jù)庫(kù)統(tǒng)統(tǒng)加載在內(nèi)存當(dāng)中進(jìn)行操作,定期通過(guò)異步操作把數(shù)據(jù)庫(kù)數(shù)據(jù)flush到硬盤上進(jìn)行保存。
因?yàn)槭羌儍?nèi)存操作,Redis的性能非常出色,每秒可以處理超過(guò)10萬(wàn)次讀寫(xiě)操作,是已知性能最快的Key-ValueDB。
Redis的出色之處不僅僅是性能,Redis最大的魅力是支持保存多種數(shù)據(jù)
結(jié)構(gòu),此外單個(gè)value的最大限制是1GB,不像memcached只能保存1MB的數(shù)據(jù),因此Redis可以用來(lái)實(shí)現(xiàn)很多有用的功能。
比方說(shuō)用他的List來(lái)做FIFO雙向鏈表,實(shí)現(xiàn)一個(gè)輕量級(jí)的高性能消息隊(duì)列服務(wù),用他的Set可以做高性能的tag系統(tǒng)等等。
另外Redis也可以對(duì)存入的Key-Value設(shè)置expire時(shí)間,因此也可以被當(dāng)作一個(gè)功能加強(qiáng)版的memcached來(lái)用。Redis的主要缺點(diǎn)是數(shù)據(jù)庫(kù)容量受到物理內(nèi)存的限制,不能用作海量數(shù)據(jù)的高性能讀寫(xiě),因此Redis適合的場(chǎng)景主要局限在較小數(shù)據(jù)量的高性能操作和運(yùn)算上。
memcached所有的值均是簡(jiǎn)單的字符串,redis作為其替代者,支持更為豐富數(shù)據(jù)類型
Redis的速度比memcached快很多
redis可以持久化其數(shù)據(jù)
String、List、Set、SortedSet、hashes
內(nèi)存。
Remote Dictionary Server
noeviction:返回錯(cuò)誤當(dāng)內(nèi)存限制達(dá)到并且客戶端嘗試執(zhí)行會(huì)讓更多內(nèi)存被使用的命令(大部分的寫(xiě)入指令,但DEL和幾個(gè)例外)
allkeys-lru:嘗試回收最少使用的鍵(LRU),使得新添加的數(shù)據(jù)有空間存放。
volatile-lru:嘗試回收最少使用的鍵(LRU),但僅限于在過(guò)期集合的鍵,使得新添加的數(shù)據(jù)有空間存放。
allkeys-random:回收隨機(jī)的鍵使得新添加的數(shù)據(jù)有空間存放。
volatile-random:回收隨機(jī)的鍵使得新添加的數(shù)據(jù)有空間存放,但僅限于在過(guò)期集合的鍵。
volatile-ttl:回收在過(guò)期集合的鍵,并且優(yōu)先回收存活時(shí)間(TTL)較短的鍵,使得新添加的數(shù)據(jù)有空間存放。
因?yàn)槟壳癓inux版本已經(jīng)相當(dāng)穩(wěn)定,而且用戶量很大,無(wú)需開(kāi)發(fā)windows版本,反而會(huì)帶來(lái)兼容性等問(wèn)題。
512M
Redis為了達(dá)到最快的讀寫(xiě)速度將數(shù)據(jù)都讀到內(nèi)存中,并通過(guò)異步的方式將數(shù)據(jù)寫(xiě)入磁盤。
所以redis具有快速和數(shù)據(jù)持久化的特征。如果不將數(shù)據(jù)放在內(nèi)存中,磁盤I/O速度為嚴(yán)重影響redis的性能。
在內(nèi)存越來(lái)越便宜的今天,redis將會(huì)越來(lái)越受歡迎。如果設(shè)置了最大使用的內(nèi)存,則數(shù)據(jù)已有記錄數(shù)達(dá)到內(nèi)存限值后不能繼續(xù)插入新值。
codis。
目前用的最多的集群方案,基本和twemproxy一致的效果,但它支持在節(jié)點(diǎn)數(shù)量改變情況下,舊節(jié)點(diǎn)數(shù)據(jù)可恢復(fù)到新hash節(jié)點(diǎn)。
rediscluster3.0自帶的集群,特點(diǎn)在于他的分布式算法不是一致性hash,而是hash槽的概念,以及自身支持節(jié)點(diǎn)設(shè)置從節(jié)點(diǎn)。具體看官方文檔介紹。
在業(yè)務(wù)代碼層實(shí)現(xiàn),起幾個(gè)毫無(wú)關(guān)聯(lián)的redis實(shí)例,在代碼層,對(duì)key進(jìn)行hash計(jì)算,然后去對(duì)應(yīng)的redis實(shí)例操作數(shù)據(jù)。這種方式對(duì)hash層代碼要求比較高,考慮部分包括,節(jié)點(diǎn)失效后的替代算法方案,數(shù)據(jù)震蕩后的自動(dòng)腳本恢復(fù),實(shí)例的監(jiān)控,等等。
有A,B,C三個(gè)節(jié)點(diǎn)的集群,在沒(méi)有復(fù)制模型的情況下,如果節(jié)點(diǎn)B失敗了,那么整個(gè)集群就會(huì)以為缺少5501-11000這個(gè)范圍的槽而不可用。
redis內(nèi)存數(shù)據(jù)集大小上升到一定大小的時(shí)候,就會(huì)施行數(shù)據(jù)淘汰策略。
會(huì)話緩存(SessionCache)
最常用的一種使用Redis的情景是會(huì)話緩存(sessioncache)。用Redis緩存會(huì)話比其他存儲(chǔ)(如Memcached)的優(yōu)勢(shì)在于:Redis提供持久化。當(dāng)維護(hù)一個(gè)不是嚴(yán)格要求一致性的緩存時(shí),如果用戶的購(gòu)物車信息全部丟失,大部分人都會(huì)不高興的,現(xiàn)在,他們還會(huì)這樣嗎?
幸運(yùn)的是,隨著Redis這些年的改進(jìn),很容易找到怎么恰當(dāng)?shù)氖褂肦edis來(lái)緩存會(huì)話的文檔。甚至廣為人知的商業(yè)平臺(tái)Magento也提供Redis的插件。
全頁(yè)緩存(FPC)
除基本的會(huì)話token之外,Redis還提供很簡(jiǎn)便的FPC平臺(tái)?;氐揭恢滦詥?wèn)題,即使重啟了Redis實(shí)例,因?yàn)橛写疟P的持久化,用戶也不會(huì)看到頁(yè)面加載速度的下降,這是一個(gè)極大改進(jìn),類似PHP本地FPC。
再次以Magento為例,Magento提供一個(gè)插件來(lái)使用Redis作為全頁(yè)緩存后端。
此外,對(duì)WordPress的用戶來(lái)說(shuō),Pantheon有一個(gè)非常好的插件wp-redis,這個(gè)插件能幫助你以最快速度加載你曾瀏覽過(guò)的頁(yè)面。
隊(duì)列
Reids在內(nèi)存存儲(chǔ)引擎領(lǐng)域的一大優(yōu)點(diǎn)是提供list和set操作,這使得Redis能作為一個(gè)很好的消息隊(duì)列平臺(tái)來(lái)使用。Redis作為隊(duì)列使用的操作,就類似于本地程序語(yǔ)言(如Python)對(duì)list的push/pop操作。
如果你快速的在Google中搜索"Redisqueues",你馬上就能找到大量的開(kāi)源項(xiàng)目,這些項(xiàng)目的目的就是利用Redis創(chuàng)建非常好的后端工具,以滿足各種隊(duì)列需求。例如,Celery有一個(gè)后臺(tái)就是使用Redis作為broker,你可以從這里去查看。
排行榜/計(jì)數(shù)器
Redis在內(nèi)存中對(duì)數(shù)字進(jìn)行遞增或遞減的操作實(shí)現(xiàn)的非常好。集合(Set)和有序集合(SortedSet)也使得我們?cè)趫?zhí)行這些操作的時(shí)候變的非常簡(jiǎn)單,Redis只是正好提供了這兩種數(shù)據(jù)結(jié)構(gòu)。所以,我們要從排序集合中獲取到排名最靠前的10個(gè)用戶-我們稱之為“user_scores”,我們只需要像下面一樣執(zhí)行即可:
當(dāng)然,這是假定你是根據(jù)你用戶的分?jǐn)?shù)做遞增的排序。如果你想返回用戶及用戶的分?jǐn)?shù),你需要這樣執(zhí)行:
ZRANGEuser_scores010WITHSCORES
AgoraGames就是一個(gè)很好的例子,用Ruby實(shí)現(xiàn)的,它的排行榜就是使用Redis來(lái)存儲(chǔ)數(shù)據(jù)的,你可以在這里看到。
發(fā)布/訂閱
最后(但肯定不是最不重要的)是Redis的發(fā)布/訂閱功能。發(fā)布/訂閱的使用場(chǎng)景確實(shí)非常多。我已看見(jiàn)人們?cè)谏缃痪W(wǎng)絡(luò)連接中使用,還可作為基于發(fā)布/訂閱的腳本觸發(fā)器,甚至用Redis的發(fā)布/訂閱功能來(lái)建立聊天系統(tǒng)!
Redisson、Jedis、lettuce等等,官方推薦使用Redisson。
Redisson是一個(gè)高級(jí)的分布式協(xié)調(diào)Redis客服端,能幫助用戶在分布式環(huán)境中輕松實(shí)現(xiàn)一些Java的對(duì)象(Bloomfilter,BitSet,Set,SetMultimap,ScoredSortedSet,SortedSet,Map,ConcurrentMap,List,ListMultimap,Queue,BlockingQueue,Deque,BlockingDeque,Semaphore,Lock,ReadWriteLock,AtomicLong,CountDownLatch,Publish/Subscribe,HyperLogLog)。
Jedis是Redis的Java實(shí)現(xiàn)的客戶端,其API提供了比較全面的Redis命令的支持;
Redisson實(shí)現(xiàn)了分布式和可擴(kuò)展的Java數(shù)據(jù)結(jié)構(gòu),和Jedis相比,功能較為簡(jiǎn)單,不支持字符串操作,不支持排序、事務(wù)、管道、分區(qū)等Redis特性。Redisson的宗旨是促進(jìn)使用者對(duì)Redis的關(guān)注分離,從而讓使用者能夠?qū)⒕Ω械胤旁谔幚順I(yè)務(wù)邏輯上。
設(shè)置密碼:config set require pass 123456 授權(quán)密碼:auth223456
Redis集群沒(méi)有使用一致性hash,而是引入了哈希槽的概念,Redis集群有16384個(gè)哈希槽,每個(gè)key通過(guò)CRC16校驗(yàn)后對(duì)16384取模來(lái)決定放置哪個(gè)槽,集群的每個(gè)節(jié)點(diǎn)負(fù)責(zé)一部分hash槽。
為了使在部分節(jié)點(diǎn)失敗或者大部分節(jié)點(diǎn)無(wú)法通信的情況下集群仍然可用,所以集群使用了主從復(fù)制模型,每個(gè)節(jié)點(diǎn)都會(huì)有N-1個(gè)復(fù)制品.
Redis并不能保證數(shù)據(jù)的強(qiáng)一致性,這意味這在實(shí)際中集群在特定的條件下可能會(huì)丟失寫(xiě)操作。
異步復(fù)制
16384個(gè)。
Redis集群目前無(wú)法做數(shù)據(jù)庫(kù)選擇,默認(rèn)在0數(shù)據(jù)庫(kù)。
ping
一次請(qǐng)求/響應(yīng)服務(wù)器能實(shí)現(xiàn)處理新的請(qǐng)求即使舊的請(qǐng)求還未被響應(yīng)。這樣就可以將多個(gè)命令發(fā)送到服務(wù)器,而不用等待回復(fù),最后在一個(gè)步驟中讀取該答復(fù)。
這就是管道(pipelining),是一種幾十年來(lái)廣泛使用的技術(shù)。例如許多POP3協(xié)議已經(jīng)實(shí)現(xiàn)支持這個(gè)功能,大大加快了從服務(wù)器下載新郵件的過(guò)程。
事務(wù)是一個(gè)單獨(dú)的隔離操作:事務(wù)中的所有命令都會(huì)序列化、按順序地執(zhí)行。事務(wù)在執(zhí)行的過(guò)程中,不會(huì)被其他客戶端發(fā)送來(lái)的命令請(qǐng)求所打斷。事務(wù)是一個(gè)原子操作:事務(wù)中的命令要么全部被執(zhí)行,要么全部都不執(zhí)行。
MULTI、EXEC、DISCARD、WATCH
EXPIRE和PERSIST命令。
盡可能使用散列表(hashes),散列表(是說(shuō)散列表里面存儲(chǔ)的數(shù)少)使用的內(nèi)存非常小,所以你應(yīng)該盡可能的將你的數(shù)據(jù)模型抽象到一個(gè)散列表里面。比如你的web系統(tǒng)中有一個(gè)用戶對(duì)象,不要為這個(gè)用戶的名稱,姓氏,郵箱,密碼設(shè)置單獨(dú)的key,而是應(yīng)該把這個(gè)用戶的所有信息存儲(chǔ)到一張散列表里面。
一個(gè)客戶端運(yùn)行了新的命令,添加了新的數(shù)據(jù)。
Redi檢查內(nèi)存使用情況,如果大于maxmemory的限制,則根據(jù)設(shè)定好的策略進(jìn)行回收。一個(gè)新的命令被執(zhí)行,等等。
所以我們不斷地穿越內(nèi)存限制的邊界,通過(guò)不斷達(dá)到邊界然后不斷地回收回到邊界以下。
如果一個(gè)命令的結(jié)果導(dǎo)致大量?jī)?nèi)存被使用(例如很大的集合的交集保存到一個(gè)新的鍵),不用多久內(nèi)存限制就會(huì)被這個(gè)內(nèi)存使用量超越。
關(guān)于“關(guān)于redis緩存的面試題有哪些”這篇文章就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,使各位可以學(xué)到更多知識(shí),如果覺(jué)得文章不錯(cuò),請(qǐng)把它分享出去讓更多的人看到。