Redis?面試題全面總結(jié)殖妇,建議收藏刁笙。

Redis 持久化機(jī)制

緩存雪崩、緩存穿透谦趣、緩存預(yù)熱疲吸、緩存更新、緩存降級(jí)等問題

熱點(diǎn)數(shù)據(jù)和冷數(shù)據(jù)是什么

Memcache與Redis的區(qū)別都有哪些前鹅?

單線程的redis為什么這么快

redis的數(shù)據(jù)類型摘悴,以及每種數(shù)據(jù)類型的使用場(chǎng)景

Redis 內(nèi)部結(jié)構(gòu)

redis的過期策略以及內(nèi)存淘汰機(jī)制

Redis 為什么是單線程的

Redis 集群方案應(yīng)該怎么做?都有哪些方案舰绘?

有沒有嘗試進(jìn)行多機(jī)redis 的部署蹂喻?如何保證數(shù)據(jù)一致的?

對(duì)于大量的請(qǐng)求怎么樣處理

Redis 常見性能問題和解決方案捂寿?

講解下Redis線程模型

為什么Redis的操作是原子性的口四,怎么保證原子性的?

Redis事務(wù)

Redis實(shí)現(xiàn)分布式鎖

Redis 持久化機(jī)制

Redis是一個(gè)支持持久化的內(nèi)存數(shù)據(jù)庫秦陋,通過持久化機(jī)制把內(nèi)存中的數(shù)據(jù)同步到硬盤文件來保證數(shù)據(jù)持久化蔓彩。當(dāng)Redis重啟后通過把硬盤文件重新加載到內(nèi)存,就能達(dá)到恢復(fù)數(shù)據(jù)的目的驳概。

實(shí)現(xiàn):?jiǎn)为?dú)創(chuàng)建fork()一個(gè)子進(jìn)程赤嚼,將當(dāng)前父進(jìn)程的數(shù)據(jù)庫數(shù)據(jù)復(fù)制到子進(jìn)程的內(nèi)存中,然后由子進(jìn)程寫入到臨時(shí)文件中顺又,持久化的過程結(jié)束了更卒,再用這個(gè)臨時(shí)文件替換上次的快照文件,然后子進(jìn)程退出稚照,內(nèi)存釋放蹂空。

RDB是Redis默認(rèn)的持久化方式。按照一定的時(shí)間周期策略把內(nèi)存的數(shù)據(jù)以快照的形式保存到硬盤的二進(jìn)制文件果录。即Snapshot快照存儲(chǔ)腌闯,對(duì)應(yīng)產(chǎn)生的數(shù)據(jù)文件為dump.rdb,通過配置文件中的save參數(shù)來定義快照的周期雕憔。( 快照可以是其所表示的數(shù)據(jù)的一個(gè)副本姿骏,也可以是數(shù)據(jù)的一個(gè)復(fù)制品。) AOF:Redis會(huì)將每一個(gè)收到的寫命令都通過Write函數(shù)追加到文件最后斤彼,類似于MySQL的binlog分瘦。當(dāng)Redis重啟是會(huì)通過重新執(zhí)行文件中保存的寫命令來在內(nèi)存中重建整個(gè)數(shù)據(jù)庫的內(nèi)容蘸泻。

當(dāng)兩種方式同時(shí)開啟時(shí),數(shù)據(jù)恢復(fù)Redis會(huì)優(yōu)先選擇AOF恢復(fù)嘲玫。

緩存雪崩悦施、緩存穿透、緩存預(yù)熱去团、緩存更新抡诞、緩存降級(jí)等問題

緩存雪崩

緩存雪崩我們可以簡(jiǎn)單的理解為:由于原有緩存失效,新緩存未到期間

(例如:我們?cè)O(shè)置緩存時(shí)采用了相同的過期時(shí)間土陪,在同一時(shí)刻出現(xiàn)大面積的緩存過期)昼汗,所有原本應(yīng)該訪問緩存的請(qǐng)求都去查詢數(shù)據(jù)庫了,而對(duì)數(shù)據(jù)庫CPU和內(nèi)存造成巨大壓力鬼雀,嚴(yán)重的會(huì)造成數(shù)據(jù)庫宕機(jī)顷窒。從而形成一系列連鎖反應(yīng),造成整個(gè)系統(tǒng)崩潰源哩。

解決辦法:

大多數(shù)系統(tǒng)設(shè)計(jì)者考慮用加鎖( 最多的解決方案)或者隊(duì)列的方式保證來保證不會(huì)有大量的線程對(duì)數(shù)據(jù)庫一次性進(jìn)行讀寫鞋吉,從而避免失效時(shí)大量的并發(fā)請(qǐng)求落到底層存儲(chǔ)系統(tǒng)上。還有一個(gè)簡(jiǎn)單方案就時(shí)講緩存失效時(shí)間分散開励烦。

緩存穿透

緩存穿透是指用戶查詢數(shù)據(jù)谓着,在數(shù)據(jù)庫沒有穷吮,自然在緩存中也不會(huì)有彤枢。這樣就導(dǎo)致用戶查詢的時(shí)候骗奖,在緩存中找不到确封,每次都要去數(shù)據(jù)庫再查詢一遍,然后返回空(相當(dāng)于進(jìn)行了兩次無用的查詢)庶柿。這樣請(qǐng)求就繞過緩存直接查數(shù)據(jù)庫,這也是經(jīng)常提的緩存命中率問題。

解決辦法:

最常見的則是采用布隆過濾器系瓢,將所有可能存在的數(shù)據(jù)哈希到一個(gè)足夠大的bitmap中,一個(gè)一定不存在的數(shù)據(jù)會(huì)被這個(gè)bitmap攔截掉句灌,從而避免了對(duì)底層存儲(chǔ)系統(tǒng)的查詢壓力夷陋。

另外也有一個(gè)更為簡(jiǎn)單粗暴的方法,如果一個(gè)查詢返回的數(shù)據(jù)為空(不管是數(shù)據(jù)不存在胰锌,還是系統(tǒng)故障)骗绕,我們?nèi)匀话堰@個(gè)空結(jié)果進(jìn)行緩存,但它的過期時(shí)間會(huì)很短资昧,最長(zhǎng)不超過五分鐘酬土。通過這個(gè)直接設(shè)置的默認(rèn)值存放到緩存,這樣第二次到緩沖中獲取就有值了格带,而不會(huì)繼續(xù)訪問數(shù)據(jù)庫撤缴,這種辦法最簡(jiǎn)單粗暴刹枉。

5TB的硬盤上放滿了數(shù)據(jù),請(qǐng)寫一個(gè)算法將這些數(shù)據(jù)進(jìn)行排重屈呕。如果這些數(shù)據(jù)是一些32bit大小的數(shù)據(jù)該如何解決微宝?如果是64bit的呢?

對(duì)于空間的利用到達(dá)了一種極致虎眨,那就是Bitmap和布隆過濾器(Bloom Filter)蟋软。

Bitmap:典型的就是哈希表

缺點(diǎn)是,Bitmap對(duì)于每個(gè)元素只能記錄1bit信息嗽桩,如果還想完成額外的功能岳守,恐怕只能靠犧牲更多的空間、時(shí)間來完成了涤躲。

布隆過濾器(推薦)

就是引入了k(k>1)k(k>1)個(gè)相互獨(dú)立的哈希函數(shù)棺耍,保證在給定的空間、誤判率下种樱,完成元素判重的過程蒙袍。

它的優(yōu)點(diǎn)是空間效率和查詢時(shí)間都遠(yuǎn)遠(yuǎn)超過一般的算法,缺點(diǎn)是有一定的誤識(shí)別率和刪除困難嫩挤。

Bloom-Filter算法的核心思想就是利用多個(gè)不同的Hash函數(shù)來解決“沖突”害幅。

Hash存在一個(gè)沖突(碰撞)的問題,用同一個(gè)Hash得到的兩個(gè)URL的值有可能相同岂昭。為了減少?zèng)_突以现,我們可以多引入幾個(gè)Hash,如果通過其中的一個(gè)Hash值我們得出某元素不在集合中约啊,那么該元素肯定不在集合中邑遏。只有在所有的Hash函數(shù)告訴我們?cè)撛卦诩现袝r(shí),才能確定該元素存在于集合中恰矩。這便是Bloom-Filter的基本思想记盒。

Bloom-Filter一般用于在大數(shù)據(jù)量的集合中判定某元素是否存在。

緩存穿透與緩存擊穿的區(qū)別

緩存擊穿:是指一個(gè)key非常熱點(diǎn)外傅,在不停的扛著大并發(fā)纪吮,大并發(fā)集中對(duì)這一個(gè)點(diǎn)進(jìn)行訪問,當(dāng)這個(gè)key在失效的瞬間萎胰,持續(xù)的大并發(fā)就穿破緩存碾盟,直接請(qǐng)求數(shù)據(jù)。

解決方案:在訪問key之前技竟,采用SETNX(set if not exists)來設(shè)置另一個(gè)短期key來鎖住當(dāng)前key的訪問冰肴,訪問結(jié)束再刪除該短期key。

給一個(gè)我公司處理的案例:背景雙機(jī)拿token,token在存一份到redis嚼沿,保證系統(tǒng)在token過期時(shí)都只有一個(gè)線程去獲取token;線上環(huán)境有兩臺(tái)機(jī)器估盘,故使用分布式鎖實(shí)現(xiàn)。

三骡尽、緩存預(yù)熱

緩存預(yù)熱這個(gè)應(yīng)該是一個(gè)比較常見的概念遣妥,相信很多小伙伴都應(yīng)該可以很容易的理解,緩存預(yù)熱就是系統(tǒng)上線后攀细,將相關(guān)的緩存數(shù)據(jù)直接加載到緩存系統(tǒng)箫踩。這樣就可以避免在用戶請(qǐng)求的時(shí)候,先查詢數(shù)據(jù)庫谭贪,然后再將數(shù)據(jù)緩存的問題境钟!用戶直接查詢事先被預(yù)熱的緩存數(shù)據(jù)!

解決思路:

直接寫個(gè)緩存刷新頁面俭识,上線時(shí)手工操作下慨削;

數(shù)據(jù)量不大,可以在項(xiàng)目啟動(dòng)的時(shí)候自動(dòng)進(jìn)行加載套媚;

定時(shí)刷新緩存缚态;

四、緩存更新

除了緩存服務(wù)器自帶的緩存失效策略之外(Redis默認(rèn)的有6中策略可供選擇)堤瘤,我們還可以根據(jù)具體的業(yè)務(wù)需求進(jìn)行自定義的緩存淘汰玫芦,常見的策略有兩種:

定時(shí)去清理過期的緩存;

當(dāng)有用戶請(qǐng)求過來時(shí)本辐,再判斷這個(gè)請(qǐng)求所用到的緩存是否過期桥帆,過期的話就去底層系統(tǒng)得到新數(shù)據(jù)并更新緩存。

兩者各有優(yōu)劣慎皱,第一種的缺點(diǎn)是維護(hù)大量緩存的key是比較麻煩的老虫,第二種的缺點(diǎn)就是每次用戶請(qǐng)求過來都要判斷緩存失效,邏輯相對(duì)比較復(fù)雜茫多!具體用哪種方案祈匙,大家可以根據(jù)自己的應(yīng)用場(chǎng)景來權(quán)衡。

五地梨、緩存降級(jí)

當(dāng)訪問量劇增、服務(wù)出現(xiàn)問題(如響應(yīng)時(shí)間慢或不響應(yīng))或非核心服務(wù)影響到核心流程的性能時(shí)缔恳,仍然需要保證服務(wù)還是可用的宝剖,即使是有損服務(wù)。系統(tǒng)可以根據(jù)一些關(guān)鍵數(shù)據(jù)進(jìn)行自動(dòng)降級(jí)歉甚,也可以配置開關(guān)實(shí)現(xiàn)人工降級(jí)万细。

降級(jí)的最終目的是保證核心服務(wù)可用,即使是有損的。而且有些服務(wù)是無法降級(jí)的(如加入購物車赖钞、結(jié)算)腰素。

以參考日志級(jí)別設(shè)置預(yù)案:

一般:比如有些服務(wù)偶爾因?yàn)榫W(wǎng)絡(luò)抖動(dòng)或者服務(wù)正在上線而超時(shí),可以自動(dòng)降級(jí)雪营;

警告:有些服務(wù)在一段時(shí)間內(nèi)成功率有波動(dòng)(如在95~100%之間)弓千,可以自動(dòng)降級(jí)或人工降級(jí),并發(fā)送告警献起;

錯(cuò)誤:比如可用率低于90%洋访,或者數(shù)據(jù)庫連接池被打爆了,或者訪問量突然猛增到系統(tǒng)能承受的最大閥值谴餐,此時(shí)可以根據(jù)情況自動(dòng)降級(jí)或者人工降級(jí)姻政;

嚴(yán)重錯(cuò)誤:比如因?yàn)樘厥庠驍?shù)據(jù)錯(cuò)誤了,此時(shí)需要緊急人工降級(jí)岂嗓。

服務(wù)降級(jí)的目的汁展,是為了防止Redis服務(wù)故障,導(dǎo)致數(shù)據(jù)庫跟著一起發(fā)生雪崩問題厌殉。因此食绿,對(duì)于不重要的緩存數(shù)據(jù),可以采取服務(wù)降級(jí)策略年枕,例如一個(gè)比較常見的做法就是炫欺,Redis出現(xiàn)問題,不去數(shù)據(jù)庫查詢熏兄,而是直接返回默認(rèn)值給用戶品洛。

熱點(diǎn)數(shù)據(jù)和冷數(shù)據(jù)是什么

熱點(diǎn)數(shù)據(jù),緩存才有價(jià)值

對(duì)于冷數(shù)據(jù)而言摩桶,大部分?jǐn)?shù)據(jù)可能還沒有再次訪問到就已經(jīng)被擠出內(nèi)存桥状,不僅占用內(nèi)存,而且價(jià)值不大硝清。頻繁修改的數(shù)據(jù)辅斟,看情況考慮使用緩存

對(duì)于上面兩個(gè)例子,壽星列表芦拿、導(dǎo)航信息都存在一個(gè)特點(diǎn)士飒,就是信息修改頻率不高,讀取通常非常高的場(chǎng)景蔗崎。

對(duì)于熱點(diǎn)數(shù)據(jù)酵幕,比如我們的某IM產(chǎn)品,生日祝福模塊缓苛,當(dāng)天的壽星列表芳撒,緩存以后可能讀取數(shù)十萬次。再舉個(gè)例子,某導(dǎo)航產(chǎn)品笔刹,我們將導(dǎo)航信息芥备,緩存以后可能讀取數(shù)百萬次。

數(shù)據(jù)更新前至少讀取兩次舌菜, 緩存才有意義萌壳。這個(gè)是最基本的策略,如果緩存還沒有起作用就失效了酷师,那就沒有太大價(jià)值了讶凉。

那存不存在,修改頻率很高山孔,但是又不得不考慮緩存的場(chǎng)景呢懂讯?有!比如台颠,這個(gè)讀取接口對(duì)數(shù)據(jù)庫的壓力很大褐望,但是又是熱點(diǎn)數(shù)據(jù),這個(gè)時(shí)候就需要考慮通過緩存手段串前,減少數(shù)據(jù)庫的壓力瘫里,比如我們的某助手產(chǎn)品的,點(diǎn)贊數(shù)荡碾,收藏?cái)?shù)谨读,分享數(shù)等是非常典型的熱點(diǎn)數(shù)據(jù),但是又不斷變化坛吁,此時(shí)就需要將數(shù)據(jù)同步保存到Redis緩存劳殖,減少數(shù)據(jù)庫壓力。

Memcache與Redis的區(qū)別都有哪些拨脉?

1)哆姻、存儲(chǔ)方式 Memecache把數(shù)據(jù)全部存在內(nèi)存之中,斷電后會(huì)掛掉玫膀,數(shù)據(jù)不能超過內(nèi)存大小矛缨。Redis有部份存在硬盤上,redis可以持久化其數(shù)據(jù)

2)帖旨、數(shù)據(jù)支持類型 memcached所有的值均是簡(jiǎn)單的字符串箕昭,redis作為其替代者,支持更為豐富的數(shù)據(jù)類型 解阅,提供list落竹,set,zset瓮钥,hash等數(shù)據(jù)結(jié)構(gòu)的存儲(chǔ)

3)筋量、使用底層模型不同 它們之間底層實(shí)現(xiàn)方式 以及與客戶端之間通信的應(yīng)用協(xié)議不一樣。Redis直接自己構(gòu)建了VM 機(jī)制 碉熄,因?yàn)橐话愕南到y(tǒng)調(diào)用系統(tǒng)函數(shù)的話桨武,會(huì)浪費(fèi)一定的時(shí)間去移動(dòng)和請(qǐng)求。

4). value 值大小不同:Redis 最大可以達(dá)到 512M锈津;memcache 只有 1mb呀酸。

5)redis的速度比memcached快很多

6)Redis支持?jǐn)?shù)據(jù)的備份,即master-slave模式的數(shù)據(jù)備份琼梆。

單線程的redis為什么這么快

(一)純內(nèi)存操作

(二)單線程操作性誉,避免了頻繁的上下文切換

(三)采用了非阻塞I/O多路復(fù)用機(jī)制

redis的數(shù)據(jù)類型,以及每種數(shù)據(jù)類型的使用場(chǎng)景

回答:一共五種

(一)String

這個(gè)其實(shí)沒啥好說的茎杂,最常規(guī)的set/get操作错览,value可以是String也可以是數(shù)字。一般做一些復(fù)雜的計(jì)數(shù)功能的緩存煌往。

(二)hash

這里value存放的是結(jié)構(gòu)化的對(duì)象倾哺,比較方便的就是操作其中的某個(gè)字段。博主在做單點(diǎn)登錄的時(shí)候刽脖,就是用這種數(shù)據(jù)結(jié)構(gòu)存儲(chǔ)用戶信息羞海,以cookieId作為key,設(shè)置30分鐘為緩存過期時(shí)間曲管,能很好的模擬出類似session的效果却邓。

(三)list

使用List的數(shù)據(jù)結(jié)構(gòu),可以做簡(jiǎn)單的消息隊(duì)列的功能院水。另外還有一個(gè)就是腊徙,可以利用lrange命令,做基于redis的分頁功能衙耕,性能極佳昧穿,用戶體驗(yàn)好。本人還用一個(gè)場(chǎng)景橙喘,很合適—取行情信息时鸵。就也是個(gè)生產(chǎn)者和消費(fèi)者的場(chǎng)景。LIST可以很好的完成排隊(duì)厅瞎,先進(jìn)先出的原則饰潜。

(四)set

因?yàn)閟et堆放的是一堆不重復(fù)值的集合。所以可以做全局去重的功能和簸。為什么不用JVM自帶的Set進(jìn)行去重彭雾?因?yàn)槲覀兊南到y(tǒng)一般都是集群部署,使用JVM自帶的Set锁保,比較麻煩薯酝,難道為了一個(gè)做一個(gè)全局去重半沽,再起一個(gè)公共服務(wù),太麻煩了吴菠。

另外者填,就是利用交集、并集做葵、差集等操作占哟,可以計(jì)算共同喜好,全部的喜好酿矢,自己獨(dú)有的喜好等功能榨乎。

(五)sorted set

sorted set多了一個(gè)權(quán)重參數(shù)score,集合中的元素能夠按score進(jìn)行排列√笨穑可以做排行榜應(yīng)用蜜暑,取TOP N操作。

Redis 內(nèi)部結(jié)構(gòu)

dict 本質(zhì)上是為了解決算法中的查找問題(Searching)是一個(gè)用于維護(hù)key和value映射關(guān)系的數(shù)據(jù)結(jié)構(gòu)策肝,與很多語言中的Map或dictionary類似史煎。本質(zhì)上是為了解決算法中的查找問題(Searching)

sds sds就等同于char * 它可以存儲(chǔ)任意二進(jìn)制數(shù)據(jù),不能像C語言字符串那樣以字符’\0’來標(biāo)識(shí)字符串的結(jié) 束驳糯,因此它必然有個(gè)長(zhǎng)度字段篇梭。

skiplist (跳躍表) 跳表是一種實(shí)現(xiàn)起來很簡(jiǎn)單,單層多指針的鏈表酝枢,它查找效率很高恬偷,堪比優(yōu)化過的二叉平衡樹,且比平衡樹的實(shí)現(xiàn)帘睦,

quicklist

ziplist 壓縮表 ziplist是一個(gè)編碼后的列表袍患,是由一系列特殊編碼的連續(xù)內(nèi)存塊組成的順序型數(shù)據(jù)結(jié)構(gòu),

redis的過期策略以及內(nèi)存淘汰機(jī)制

redis采用的是定期刪除+惰性刪除策略竣付。

為什么不用定時(shí)刪除策略?

定時(shí)刪除,用一個(gè)定時(shí)器來負(fù)責(zé)監(jiān)視key,過期則自動(dòng)刪除诡延。雖然內(nèi)存及時(shí)釋放,但是十分消耗CPU資源古胆。在大并發(fā)請(qǐng)求下肆良,CPU要將時(shí)間應(yīng)用在處理請(qǐng)求,而不是刪除key,因此沒有采用這一策略.

定期刪除+惰性刪除是如何工作的呢?

定期刪除逸绎,redis默認(rèn)每個(gè)100ms檢查惹恃,是否有過期的key,有過期key則刪除。需要說明的是棺牧,redis不是每個(gè)100ms將所有的key檢查一次巫糙,而是隨機(jī)抽取進(jìn)行檢查(如果每隔100ms,全部key進(jìn)行檢查,redis豈不是卡死)颊乘。因此参淹,如果只采用定期刪除策略醉锄,會(huì)導(dǎo)致很多key到時(shí)間沒有刪除。

于是浙值,惰性刪除派上用場(chǎng)榆鼠。也就是說在你獲取某個(gè)key的時(shí)候,redis會(huì)檢查一下亥鸠,這個(gè)key如果設(shè)置了過期時(shí)間那么是否過期了?如果過期了此時(shí)就會(huì)刪除识啦。

采用定期刪除+惰性刪除就沒其他問題了么?

不是的负蚊,如果定期刪除沒刪除key。然后你也沒即時(shí)去請(qǐng)求key颓哮,也就是說惰性刪除也沒生效家妆。這樣,redis的內(nèi)存會(huì)越來越高冕茅。那么就應(yīng)該采用內(nèi)存淘汰機(jī)制伤极。

在redis.conf中有一行配置

maxmemory-policy volatile-lru1

該配置就是配內(nèi)存淘汰策略的(什么,你沒配過姨伤?好好反省一下自己)

volatile-lru:從已設(shè)置過期時(shí)間的數(shù)據(jù)集(server.db[i].expires)中挑選最近最少使用的數(shù)據(jù)淘汰

volatile-ttl:從已設(shè)置過期時(shí)間的數(shù)據(jù)集(server.db[i].expires)中挑選將要過期的數(shù)據(jù)淘汰

volatile-random:從已設(shè)置過期時(shí)間的數(shù)據(jù)集(server.db[i].expires)中任意選擇數(shù)據(jù)淘汰

allkeys-lru:從數(shù)據(jù)集(server.db[i].dict)中挑選最近最少使用的數(shù)據(jù)淘汰

allkeys-random:從數(shù)據(jù)集(server.db[i].dict)中任意選擇數(shù)據(jù)淘汰

no-enviction(驅(qū)逐):禁止驅(qū)逐數(shù)據(jù)哨坪,新寫入操作會(huì)報(bào)錯(cuò)

ps:如果沒有設(shè)置 expire 的key, 不滿足先決條件(prerequisites); 那么 volatile-lru, volatile-random 和 volatile-ttl 策略的行為, 和 noeviction(不刪除) 基本上一致。

Redis 為什么是單線程的

官方FAQ表示乍楚,因?yàn)镽edis是基于內(nèi)存的操作当编,CPU不是Redis的瓶頸,Redis的瓶頸最有可能是機(jī)器內(nèi)存的大小或者網(wǎng)絡(luò)帶寬徒溪。

既然單線程容易實(shí)現(xiàn)忿偷,而且CPU不會(huì)成為瓶頸,那就順理成章地采用單線程的方案了(畢竟采用多線程會(huì)有很多麻煩k凇)Redis利用隊(duì)列技術(shù)將并發(fā)訪問變?yōu)榇性L問

1)絕大部分請(qǐng)求是純粹的內(nèi)存操作(非忱鹎牛快速)

2)采用單線程,避免了不必要的上下文切換和競(jìng)爭(zhēng)條件

3)非阻塞IO優(yōu)點(diǎn):

速度快,因?yàn)閿?shù)據(jù)存在內(nèi)存中渠概,類似于HashMap茶凳,HashMap的優(yōu)勢(shì)就是查找和操作的時(shí)間復(fù)雜度都是O(1)

支持豐富數(shù)據(jù)類型,支持string播揪,list慧妄,set,sorted set剪芍,hash

支持事務(wù)塞淹,操作都是原子性,所謂的原子性就是對(duì)數(shù)據(jù)的更改要么全部執(zhí)行罪裹,要么全部不執(zhí)行

豐富的特性:可用于緩存饱普,消息运挫,按key設(shè)置過期時(shí)間,過期后將會(huì)自動(dòng)刪除如何解決redis的并發(fā)競(jìng)爭(zhēng)key問題

同時(shí)有多個(gè)子系統(tǒng)去set一個(gè)key套耕。這個(gè)時(shí)候要注意什么呢谁帕?

不推薦使用redis的事務(wù)機(jī)制。因?yàn)槲覀兊纳a(chǎn)環(huán)境冯袍,基本都是redis集群環(huán)境匈挖,做了數(shù)據(jù)分片操作。你一個(gè)事務(wù)中有涉及到多個(gè)key操作的時(shí)候康愤,這多個(gè)key不一定都存儲(chǔ)在同一個(gè)redis-server上儡循。因此,redis的事務(wù)機(jī)制征冷,十分雞肋择膝。

如果對(duì)這個(gè)key操作,不要求順序:準(zhǔn)備一個(gè)分布式鎖检激,大家去搶鎖肴捉,搶到鎖就做set操作即可

如果對(duì)這個(gè)key操作,要求順序:分布式鎖+時(shí)間戳叔收。假設(shè)這會(huì)系統(tǒng)B先搶到鎖齿穗,將key1設(shè)置為{valueB 3:05}。接下來系統(tǒng)A搶到鎖饺律,發(fā)現(xiàn)自己的valueA的時(shí)間戳早于緩存中的時(shí)間戳缤灵,那就不做set操作了。以此類推蓝晒。

利用隊(duì)列腮出,將set方法變成串行訪問也可以redis遇到高并發(fā),如果保證讀寫key的一致性

對(duì)redis的操作都是具有原子性的,是線程安全的操作,你不用考慮并發(fā)問題,redis內(nèi)部已經(jīng)幫你處理好并發(fā)的問題了芝薇。

Redis 集群方案應(yīng)該怎么做胚嘲?都有哪些方案?

1.twemproxy洛二,大概概念是馋劈,它類似于一個(gè)代理方式, 使用時(shí)在本需要連接 redis 的地方改為連接 twemproxy晾嘶, 它會(huì)以一個(gè)代理的身份接收請(qǐng)求并使用一致性 hash 算法妓雾,將請(qǐng)求轉(zhuǎn)接到具體 redis,將結(jié)果再返回 twemproxy垒迂。

缺點(diǎn):twemproxy 自身單端口實(shí)例的壓力械姻,使用一致性 hash 后,對(duì) redis 節(jié)點(diǎn)數(shù)量改變時(shí)候的計(jì)算值的改變机断,數(shù)據(jù)無法自動(dòng)移動(dòng)到新的節(jié)點(diǎn)楷拳。

2.codis绣夺,目前用的最多的集群方案,基本和 twemproxy 一致的效果欢揖,但它支持在 節(jié)點(diǎn)數(shù)量改變情況下陶耍,舊節(jié)點(diǎn)數(shù)據(jù)可恢復(fù)到新 hash 節(jié)點(diǎn)

3.redis cluster3.0 自帶的集群,特點(diǎn)在于他的分布式算法不是一致性 hash她混,而是 hash 槽的概念烈钞,以及自身支持節(jié)點(diǎn)設(shè)置從節(jié)點(diǎn)。具體看官方文檔介紹坤按。

有沒有嘗試進(jìn)行多機(jī)redis 的部署毯欣?如何保證數(shù)據(jù)一致的?

主從復(fù)制晋涣,讀寫分離

一類是主數(shù)據(jù)庫(master)一類是從數(shù)據(jù)庫(slave),主數(shù)據(jù)庫可以進(jìn)行讀寫操作沉桌,當(dāng)發(fā)生寫操作的時(shí)候自動(dòng)將數(shù)據(jù)同步到從數(shù)據(jù)庫谢鹊,而從數(shù)據(jù)庫一般是只讀的,并接收主數(shù)據(jù)庫同步過來的數(shù)據(jù)留凭,一個(gè)主數(shù)據(jù)庫可以有多個(gè)從數(shù)據(jù)庫佃扼,而一個(gè)從數(shù)據(jù)庫只能有一個(gè)主數(shù)據(jù)庫。

對(duì)于大量的請(qǐng)求怎么樣處理

redis是一個(gè)單線程程序蔼夜,也就說同一時(shí)刻它只能處理一個(gè)客戶端請(qǐng)求兼耀;

redis是通過IO多路復(fù)用(select,epoll, kqueue求冷,依據(jù)不同的平臺(tái)瘤运,采取不同的實(shí)現(xiàn))來處理多個(gè)客戶端請(qǐng)求的

Redis 常見性能問題和解決方案?

(1) Master 最好不要做任何持久化工作匠题,如 RDB 內(nèi)存快照和 AOF 日志文件

(2) 如果數(shù)據(jù)比較重要拯坟,某個(gè) Slave 開啟 AOF 備份數(shù)據(jù),策略設(shè)置為每秒同步一次

(3) 為了主從復(fù)制的速度和連接的穩(wěn)定性韭山, Master 和 Slave 最好在同一個(gè)局域網(wǎng)內(nèi)

(4) 盡量避免在壓力很大的主庫上增加從庫

(5) 主從復(fù)制不要用圖狀結(jié)構(gòu)郁季,用單向鏈表結(jié)構(gòu)更為穩(wěn)定,即:Master <- Slave1 <- Slave2 <- Slave3…

講解下Redis線程模型

文件事件處理器包括分別是套接字钱磅、 I/O 多路復(fù)用程序梦裂、 文件事件分派器(dispatcher)、 以及事件處理器盖淡。使用 I/O 多路復(fù)用程序來同時(shí)監(jiān)聽多個(gè)套接字年柠, 并根據(jù)套接字目前執(zhí)行的任務(wù)來為套接字關(guān)聯(lián)不同的事件處理器。

當(dāng)被監(jiān)聽的套接字準(zhǔn)備好執(zhí)行連接應(yīng)答(accept)褪迟、讀缺肷肌(read)毅往、寫入(write)、關(guān)閉(close)等操作時(shí)派近, 與操作相對(duì)應(yīng)的文件事件就會(huì)產(chǎn)生攀唯, 這時(shí)文件事件處理器就會(huì)調(diào)用套接字之前關(guān)聯(lián)好的事件處理器來處理這些事件。

I/O 多路復(fù)用程序負(fù)責(zé)監(jiān)聽多個(gè)套接字渴丸, 并向文件事件分派器傳送那些產(chǎn)生了事件的套接字侯嘀。

工作原理:

I/O 多路復(fù)用程序負(fù)責(zé)監(jiān)聽多個(gè)套接字, 并向文件事件分派器傳送那些產(chǎn)生了事件的套接字谱轨。

盡管多個(gè)文件事件可能會(huì)并發(fā)地出現(xiàn)戒幔, 但 I/O 多路復(fù)用程序總是會(huì)將所有產(chǎn)生事件的套接字都入隊(duì)到一個(gè)隊(duì)列里面, 然后通過這個(gè)隊(duì)列土童, 以有序(sequentially)诗茎、同步(synchronously)、每次一個(gè)套接字的方式向文件事件分派器傳送套接字:

當(dāng)上一個(gè)套接字產(chǎn)生的事件被處理完畢之后(該套接字為事件所關(guān)聯(lián)的事件處理器執(zhí)行完畢)献汗, I/O 多路復(fù)用程序才會(huì)繼續(xù)向文件事件分派器傳送下一個(gè)套接字敢订。如果一個(gè)套接字又可讀又可寫的話, 那么服務(wù)器將先讀套接字罢吃, 后寫套接字.

為什么Redis的操作是原子性的楚午,怎么保證原子性的?

對(duì)于Redis而言尿招,命令的原子性指的是:一個(gè)操作的不可以再分矾柜,操作要么執(zhí)行,要么不執(zhí)行就谜。

Redis的操作之所以是原子性的怪蔑,是因?yàn)镽edis是單線程的。(Redis新版本已經(jīng)引入多線程丧荐,這里基于舊版本的Redis)

Redis本身提供的所有API都是原子操作饮睬,Redis中的事務(wù)其實(shí)是要保證批量操作的原子性。

多個(gè)命令在并發(fā)中也是原子性的嗎篮奄?

不一定捆愁, 將get和set改成單命令操作,incr 窟却。使用Redis的事務(wù)昼丑,或者使用Redis+Lua==的方式實(shí)現(xiàn).

Redis事務(wù)

Redis事務(wù)功能是通過MULTI、EXEC夸赫、DISCARD和WATCH 四個(gè)原語實(shí)現(xiàn)的

Redis會(huì)將一個(gè)事務(wù)中的所有命令序列化菩帝,然后按順序執(zhí)行。

redis 不支持回滾“Redis 在事務(wù)失敗時(shí)不進(jìn)行回滾,而是繼續(xù)執(zhí)行余下的命令”呼奢, 所以 Redis 的內(nèi)部可以保持簡(jiǎn)單且快速宜雀。

如果在一個(gè)事務(wù)中的命令出現(xiàn)錯(cuò)誤,那么所有的命令都不會(huì)執(zhí)行握础;

如果在一個(gè)事務(wù)中出現(xiàn)運(yùn)行錯(cuò)誤辐董,那么正確的命令會(huì)被執(zhí)行。

注:redis的discard只是結(jié)束本次事務(wù),正確命令造成的影響仍然存在.

1)MULTI命令用于開啟一個(gè)事務(wù)禀综,它總是返回OK简烘。MULTI執(zhí)行之后,客戶端可以繼續(xù)向服務(wù)器發(fā)送任意多條命令定枷,這些命令不會(huì)立即被執(zhí)行孤澎,而是被放到一個(gè)隊(duì)列中,當(dāng)EXEC命令被調(diào)用時(shí)欠窒,所有隊(duì)列中的命令才會(huì)被執(zhí)行覆旭。

2)EXEC:執(zhí)行所有事務(wù)塊內(nèi)的命令。返回事務(wù)塊內(nèi)所有命令的返回值岖妄,按命令執(zhí)行的先后順序排列型将。當(dāng)操作被打斷時(shí),返回空值 nil 衣吠。

3)通過調(diào)用DISCARD茶敏,客戶端可以清空事務(wù)隊(duì)列壤靶,并放棄執(zhí)行事務(wù)缚俏, 并且客戶端會(huì)從事務(wù)狀態(tài)中退出。

4)WATCH 命令可以為 Redis 事務(wù)提供 check-and-set (CAS)行為贮乳∮腔唬可以監(jiān)控一個(gè)或多個(gè)鍵,一旦其中有一個(gè)鍵被修改(或刪除)向拆,之后的事務(wù)就不會(huì)執(zhí)行亚茬,監(jiān)控一直持續(xù)到EXEC命令。

Redis實(shí)現(xiàn)分布式鎖

Redis為單進(jìn)程單線程模式浓恳,采用隊(duì)列模式將并發(fā)訪問變成串行訪問刹缝,且多客戶端對(duì)Redis的連接并不存在競(jìng)爭(zhēng)關(guān)系Redis中可以使用SETNX命令實(shí)現(xiàn)分布式鎖。

將 key 的值設(shè)為 value 颈将,當(dāng)且僅當(dāng) key 不存在梢夯。若給定的 key 已經(jīng)存在,則 SETNX 不做任何動(dòng)作


解鎖:使用 del key 命令就能釋放鎖

解決死鎖:

通過Redis中expire()給鎖設(shè)定最大持有時(shí)間晴圾,如果超過颂砸,則Redis來幫我們釋放鎖。

使用 setnx key “當(dāng)前系統(tǒng)時(shí)間+鎖持有的時(shí)間”和getset key “當(dāng)前系統(tǒng)時(shí)間+鎖持有的時(shí)間”組合的命令就可以實(shí)現(xiàn)。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末人乓,一起剝皮案震驚了整個(gè)濱河市勤篮,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌色罚,老刑警劉巖碰缔,帶你破解...
    沈念sama閱讀 222,627評(píng)論 6 517
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異保屯,居然都是意外死亡手负,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,180評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門姑尺,熙熙樓的掌柜王于貴愁眉苦臉地迎上來竟终,“玉大人,你說我怎么就攤上這事切蟋⊥炒罚” “怎么了?”我有些...
    開封第一講書人閱讀 169,346評(píng)論 0 362
  • 文/不壞的土叔 我叫張陵柄粹,是天一觀的道長(zhǎng)喘鸟。 經(jīng)常有香客問我,道長(zhǎng)驻右,這世上最難降的妖魔是什么什黑? 我笑而不...
    開封第一講書人閱讀 60,097評(píng)論 1 300
  • 正文 為了忘掉前任,我火速辦了婚禮堪夭,結(jié)果婚禮上愕把,老公的妹妹穿的比我還像新娘。我一直安慰自己森爽,他們只是感情好恨豁,可當(dāng)我...
    茶點(diǎn)故事閱讀 69,100評(píng)論 6 398
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著爬迟,像睡著了一般橘蜜。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上付呕,一...
    開封第一講書人閱讀 52,696評(píng)論 1 312
  • 那天计福,我揣著相機(jī)與錄音,去河邊找鬼徽职。 笑死象颖,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的活箕。 我是一名探鬼主播力麸,決...
    沈念sama閱讀 41,165評(píng)論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼可款,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了克蚂?” 一聲冷哼從身側(cè)響起闺鲸,我...
    開封第一講書人閱讀 40,108評(píng)論 0 277
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎埃叭,沒想到半個(gè)月后摸恍,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,646評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡赤屋,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,709評(píng)論 3 342
  • 正文 我和宋清朗相戀三年立镶,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片类早。...
    茶點(diǎn)故事閱讀 40,861評(píng)論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡媚媒,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出涩僻,到底是詐尸還是另有隱情缭召,我是刑警寧澤,帶...
    沈念sama閱讀 36,527評(píng)論 5 351
  • 正文 年R本政府宣布逆日,位于F島的核電站嵌巷,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏室抽。R本人自食惡果不足惜搪哪,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,196評(píng)論 3 336
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望坪圾。 院中可真熱鬧晓折,春花似錦、人聲如沸神年。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,698評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽已日。三九已至,卻和暖如春栅屏,著一層夾襖步出監(jiān)牢的瞬間飘千,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,804評(píng)論 1 274
  • 我被黑心中介騙來泰國(guó)打工栈雳, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留护奈,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 49,287評(píng)論 3 379
  • 正文 我出身青樓哥纫,卻偏偏與公主長(zhǎng)得像霉旗,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,860評(píng)論 2 361

推薦閱讀更多精彩內(nèi)容