這篇文章將為大家詳細講解有關(guān)redis中有哪些高頻面試題,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。
目前成都創(chuàng)新互聯(lián)已為上千余家的企業(yè)提供了網(wǎng)站建設(shè)、域名、虛擬空間、網(wǎng)站改版維護、企業(yè)網(wǎng)站設(shè)計、蕪湖網(wǎng)站維護等服務(wù),公司將堅持客戶導(dǎo)向、應(yīng)用為本的策略,正道將秉承"和諧、參與、激情"的文化,與客戶和合作伙伴齊心協(xié)力一起成長,共同發(fā)展。
很多人只知道是 K/V NOSQL 內(nèi)存數(shù)據(jù)庫,單線程……這都是沒有全面理解 Redis 導(dǎo)致無法繼續(xù)深問下去。
這個問題是基礎(chǔ)摸底,我們可以從 Redis 不同數(shù)據(jù)類型底層的數(shù)據(jù)結(jié)構(gòu)實現(xiàn)、完全基于內(nèi)存、IO 多路復(fù)用網(wǎng)絡(luò)模型、線程模型、漸進式 rehash…...
我們可以先說到底有多快,根據(jù)官方數(shù)據(jù),Redis 的 QPS 可以達到約 100000(每秒請求數(shù)),有興趣的可以參考官方的基準程序測試《How fast is Redis?》,地址:redis.io/topics/benc…
橫軸是連接數(shù),縱軸是 QPS。
這張圖反映了一個數(shù)量級,通過量化讓面試官覺得你有看過官方文檔,很嚴謹。
Redis 是基于內(nèi)存的數(shù)據(jù)庫,跟磁盤數(shù)據(jù)庫相比,完全吊打磁盤的速度。
不論讀寫操作都是在內(nèi)存上完成的,我們分別對比下內(nèi)存操作與磁盤操作的差異。
磁盤調(diào)用
內(nèi)存操作
內(nèi)存直接由 CPU 控制,也就是 CPU 內(nèi)部集成的內(nèi)存控制器,所以說內(nèi)存是直接與 CPU 對接,享受與 CPU 通信的最優(yōu)帶寬。
最后以一張圖量化系統(tǒng)的各種延時時間(部分數(shù)據(jù)引用 Brendan Gregg)
學(xué)習(xí) MySQL 的時候我知道為了提高檢索速度使用了 B+ Tree 數(shù)據(jù)結(jié)構(gòu),所以 Redis 速度快應(yīng)該也跟數(shù)據(jù)結(jié)構(gòu)有關(guān)。
Redis 一共有 5 種數(shù)據(jù)類型,String、List、Hash、Set、SortedSet
。
不同的數(shù)據(jù)類型底層使用了一種或者多種數(shù)據(jù)結(jié)構(gòu)來支撐,目的就是為了追求更快的速度。
碼哥寄語:我們可以分別說明每種數(shù)據(jù)類型底層的數(shù)據(jù)結(jié)構(gòu)優(yōu)點,很多人只知道數(shù)據(jù)類型,而說出底層數(shù)據(jù)結(jié)構(gòu)就能讓人眼前一亮。
SDS 中 len 保存這字符串的長度,O(1) 時間復(fù)雜度查詢字符串長度信息。
空間預(yù)分配:SDS 被修改后,程序不僅會為 SDS 分配所需要的必須空間,還會分配額外的未使用空間。
惰性空間釋放:當對 SDS 進行縮短操作時,程序并不會回收多余的內(nèi)存空間,而是使用 free 字段將這些字節(jié)數(shù)量記錄下來不釋放,后面如果需要 append 操作,則直接使用 free 中未使用的空間,減少了內(nèi)存的分配。
壓縮列表是 List 、hash、 sorted Set 三種數(shù)據(jù)類型底層實現(xiàn)之一。
當一個列表只有少量數(shù)據(jù)的時候,并且每個列表項要么就是小整數(shù)值,要么就是長度比較短的字符串,那么 Redis 就會使用壓縮列表來做列表鍵的底層實現(xiàn)。
這樣內(nèi)存緊湊,節(jié)約內(nèi)存。
后續(xù)版本對列表數(shù)據(jù)結(jié)構(gòu)進行了改造,使用 quicklist 代替了 ziplist 和 linkedlist。
quicklist 是 ziplist 和 linkedlist 的混合體,它將 linkedlist 按段切分,每一段使用 ziplist 來緊湊存儲,多個 ziplist 之間使用雙向指針串接起來。
sorted set 類型的排序功能便是通過「跳躍列表」數(shù)據(jù)結(jié)構(gòu)來實現(xiàn)。
跳躍表(skiplist)是一種有序數(shù)據(jù)結(jié)構(gòu),它通過在每個節(jié)點中維持多個指向其他節(jié)點的指針,從而達到快速訪問節(jié)點的目的。
跳表在鏈表的基礎(chǔ)上,增加了多層級索引,通過索引位置的幾個跳轉(zhuǎn),實現(xiàn)數(shù)據(jù)的快速定位,如下圖所示:
當一個集合只包含整數(shù)值元素,并且這個集合的元素數(shù)量不多時,Redis 就會使用整數(shù)集合作為集合鍵的底層實現(xiàn),節(jié)省內(nèi)存。
碼哥寄語:我們需要注意的是,Redis 的單線程指的是 Redis 的網(wǎng)絡(luò) IO (6.x 版本后網(wǎng)絡(luò) IO 使用多線程)以及鍵值對指令讀寫是由一個線程來執(zhí)行的。對于 Redis 的持久化、集群數(shù)據(jù)同步、異步刪除等都是其他線程執(zhí)行。
千萬別說 Redis 就只有一個線程。
單線程指的是 Redis 鍵值對讀寫指令的執(zhí)行是單線程。
先說官方答案,讓人覺得足夠嚴謹,而不是人云亦云去背誦一些博客。
官方答案:因為 Redis 是基于內(nèi)存的操作,CPU 不是 Redis 的瓶頸,Redis 的瓶頸最有可能是機器內(nèi)存的大小或者網(wǎng)絡(luò)帶寬。既然單線程容易實現(xiàn),而且 CPU 不會成為瓶頸,那就順理成章地采用單線程的方案了。原文地址:redis.io/topics/faq。
為啥不用多線程執(zhí)行充分利用 CPU 呢?
在運行每個任務(wù)之前,CPU 需要知道任務(wù)在何處加載并開始運行。也就是說,系統(tǒng)需要幫助它預(yù)先設(shè)置 CPU 寄存器和程序計數(shù)器,這稱為 CPU 上下文。
切換上下文時,我們需要完成一系列工作,這是非常消耗資源的操作。
引入多線程開發(fā),就需要使用同步原語來保護共享資源的并發(fā)讀寫,增加代碼復(fù)雜度和調(diào)試難度。
單線程又什么好處?
不會因為線程創(chuàng)建導(dǎo)致的性能消耗;
避免上下文切換引起的 CPU 消耗,沒有多線程切換的開銷;
避免了線程之間的競爭問題,比如添加鎖、釋放鎖、死鎖等,不需要考慮各種鎖問題。
代碼更清晰,處理邏輯簡單。
Redis 采用 I/O 多路復(fù)用技術(shù),并發(fā)處理連接。采用了 epoll + 自己實現(xiàn)的簡單的事件框架。
epoll 中的讀、寫、關(guān)閉、連接都轉(zhuǎn)化成了事件,然后利用 epoll 的多路復(fù)用特性,絕不在 IO 上浪費一點時間。
Redis 線程不會阻塞在某一個特定的監(jiān)聽或已連接套接字上,也就是說,不會阻塞在某一個特定的客戶端請求處理上。正因為此,Redis 可以同時和多個客戶端連接并處理請求,從而提升并發(fā)性。
Redis 整體就是一個 哈希表來保存所有的鍵值對,無論數(shù)據(jù)類型是 5 種的任意一種。哈希表,本質(zhì)就是一個數(shù)組,每個元素被叫做哈希桶,不管什么數(shù)據(jù)類型,每個桶里面的 entry 保存著實際具體值的指針。
而哈希表的時間復(fù)雜度是 O(1),只需要計算每個鍵的哈希值,便知道對應(yīng)的哈希桶位置,定位桶里面的 entry 找到對應(yīng)數(shù)據(jù),這個也是 Redis 快的原因之一。
Redis 使用對象(redisObject)來表示數(shù)據(jù)庫中的鍵值,當我們在 Redis 中創(chuàng)建一個鍵值對時,至少創(chuàng)建兩個對象,一個對象是用做鍵值對的鍵對象,另一個是鍵值對的值對象。
也就是每個 entry 保存著 「鍵值對」的 redisObject 對象,通過 redisObject 的指針找到對應(yīng)數(shù)據(jù)。
typedef struct redisObject{ //類型 unsigned type:4; //編碼 unsigned encoding:4; //指向底層數(shù)據(jù)結(jié)構(gòu)的指針 void *ptr; //... }robj;復(fù)制代碼
Redis 通過鏈式哈希解決沖突:也就是同一個 桶里面的元素使用鏈表保存。但是當鏈表過長就會導(dǎo)致查找性能變差可能,所以 Redis 為了追求快,使用了兩個全局哈希表。用于 rehash 操作,增加現(xiàn)有的哈希桶數(shù)量,減少哈希沖突。
開始默認使用 「hash 表 1 」保存鍵值對數(shù)據(jù),「hash 表 2」 此刻沒有分配空間。當數(shù)據(jù)越來多觸發(fā) rehash 操作,則執(zhí)行以下操作:
給 「hash 表 2 」分配更大的空間;
將 「hash 表 1 」的數(shù)據(jù)重新映射拷貝到 「hash 表 2」 中;
釋放 hash 表 1 的空間。
值得注意的是,將 hash 表 1 的數(shù)據(jù)重新映射到 hash 表 2 的過程中并不是一次性的,這樣會造成 Redis 阻塞,無法提供服務(wù)。
而是采用了漸進式 rehash,每次處理客戶端請求的時候,先從「 hash 表 1」 中第一個索引開始,將這個位置的 所有數(shù)據(jù)拷貝到 「hash 表 2」 中,就這樣將 rehash 分散到多次請求過程中,避免耗時阻塞。
Redis 的數(shù)據(jù)持久化使用了「RDB 數(shù)據(jù)快照」的方式來實現(xiàn)宕機快速恢復(fù)。但是 過于頻繁的執(zhí)行全量數(shù)據(jù)快照,有兩個嚴重性能開銷:
頻繁生成 RDB 文件寫入磁盤,磁盤壓力過大。會出現(xiàn)上一個 RDB 還未執(zhí)行完,下一個又開始生成,陷入死循環(huán)。
fork 出 bgsave 子進程會阻塞主線程,主線程的內(nèi)存越大,阻塞時間越長。
所以 Redis 還設(shè)計了 AOF 寫后日志記錄對內(nèi)存進行修改的指令記錄。
面試官:什么是 RDB 內(nèi)存快照?
在 Redis 執(zhí)行「寫」指令過程中,內(nèi)存數(shù)據(jù)會一直變化。所謂的內(nèi)存快照,指的就是 Redis 內(nèi)存中的數(shù)據(jù)在某一刻的狀態(tài)數(shù)據(jù)。
好比時間定格在某一刻,當我們拍照的,通過照片就能把某一刻的瞬間畫面完全記錄下來。
Redis 跟這個類似,就是把某一刻的數(shù)據(jù)以文件的形式拍下來,寫到磁盤上。這個快照文件叫做 RDB 文件,RDB 就是 Redis DataBase 的縮寫。
在做數(shù)據(jù)恢復(fù)時,直接將 RDB 文件讀入內(nèi)存完成恢復(fù)。
面試官:在生成 RDB 期間,Redis 可以同時處理寫請求么?
可以的,Redis 使用操作系統(tǒng)的多進程寫時復(fù)制技術(shù) COW(Copy On Write)來實現(xiàn)快照持久化,保證數(shù)據(jù)一致性。
Redis 在持久化時會調(diào)用 glibc 的函數(shù)fork
產(chǎn)生一個子進程,快照持久化完全交給子進程來處理,父進程繼續(xù)處理客戶端請求。
當主線程執(zhí)行寫指令修改數(shù)據(jù)的時候,這個數(shù)據(jù)就會復(fù)制一份副本, bgsave
子進程讀取這個副本數(shù)據(jù)寫到 RDB 文件。
這既保證了快照的完整性,也允許主線程同時對數(shù)據(jù)進行修改,避免了對正常業(yè)務(wù)的影響。
面試官:那 AOF 又是什么?
AOF 日志記錄了自 Redis 實例創(chuàng)建以來所有的修改性指令序列,那么就可以通過對一個空的 Redis 實例順序執(zhí)行所有的指令,也就是「重放」,來恢復(fù) Redis 當前實例的內(nèi)存數(shù)據(jù)結(jié)構(gòu)的狀態(tài)。
Redis 提供的 AOF 配置項appendfsync
寫回策略直接決定 AOF 持久化功能的效率和安全性。
always:同步寫回,寫指令執(zhí)行完畢立馬將 aof_buf
緩沖區(qū)中的內(nèi)容刷寫到 AOF 文件。
everysec:每秒寫回,寫指令執(zhí)行完,日志只會寫到 AOF 文件緩沖區(qū),每隔一秒就把緩沖區(qū)內(nèi)容同步到磁盤。
no:操作系統(tǒng)控制,寫執(zhí)行執(zhí)行完畢,把日志寫到 AOF 文件內(nèi)存緩沖區(qū),由操作系統(tǒng)決定何時刷寫到磁盤。
沒有兩全其美的策略,我們需要在性能和可靠性上做一個取舍。
面試官:既然 RDB 有兩個性能問題,那為何不用 AOF 即可。
AOF 寫前日志,記錄的是每個「寫」指令操作。不會像 RDB 全量快照導(dǎo)致性能損耗,但是執(zhí)行速度沒有 RDB 快,同時日志文件過大也會造成性能問題。
所以,Redis 設(shè)計了一個殺手锏「AOF 重寫機制」,Redis 提供了 bgrewriteaof
指令用于對 AOF 日志進行瘦身。
其原理就是開辟一個子進程對內(nèi)存進行遍歷轉(zhuǎn)換成一系列 Redis 的操作指令,序列化到一個新的 AOF 日志文件中。序列化完畢后再將操作期間發(fā)生的增量 AOF 日志追加到這個新的 AOF 日志文件中,追加完畢后就立即替代舊的 AOF 日志文件了,瘦身工作就完成了。
面試官:如何實現(xiàn) 數(shù)據(jù)盡可能少丟失又能兼顧性能呢?
重啟 Redis 時,我們很少使用 rdb 來恢復(fù)內(nèi)存狀態(tài),因為會丟失大量數(shù)據(jù)。我們通常使用 AOF 日志重放,但是重放 AOF 日志性能相對 rdb 來說要慢很多,這樣在 Redis 實例很大的情況下,啟動需要花費很長的時間。
Redis 4.0 為了解決這個問題,帶來了一個新的持久化選項——混合持久化。將 rdb 文件的內(nèi)容和增量的 AOF 日志文件存在一起。這里的 AOF 日志不再是全量的日志,而是自持久化開始到持久化結(jié)束的這段時間發(fā)生的增量 AOF 日志,通常這部分 AOF 日志很小。
于是在 Redis 重啟的時候,可以先加載 rdb 的內(nèi)容,然后再重放增量 AOF 日志就可以完全替代之前的 AOF 全量文件重放,重啟效率因此大幅得到提升。
Redis 提供了主從模式,通過主從復(fù)制,將數(shù)據(jù)冗余一份復(fù)制到其他 Redis 服務(wù)器。
面試官:主從之間數(shù)據(jù)如何保證一致性?
為了保證副本數(shù)據(jù)的一致性,主從架構(gòu)采用了讀寫分離的方式。
讀操作:主、從庫都可以執(zhí)行;
寫操作:主庫先執(zhí)行,之后將寫操作同步到從庫;
面試官:主從復(fù)制還有其他作用么?
故障恢復(fù):當主節(jié)點宕機,其他節(jié)點依然可以提供服務(wù);
負載均衡:Master 節(jié)點提供寫服務(wù),Slave 節(jié)點提供讀服務(wù),分擔壓力;
高可用基石:是哨兵和 cluster 實施的基礎(chǔ),是高可用的基石。
面試官:主從復(fù)制如何實現(xiàn)的?
同步分為三種情況:
第一次主從庫全量復(fù)制;
主從正常運行期間的同步;
主從庫間網(wǎng)絡(luò)斷開重連同步。
面試官:第一次同步怎么實現(xiàn)?
主從庫第一次復(fù)制過程大體可以分為 3 個階段:連接建立階段(即準備階段)、主庫同步數(shù)據(jù)到從庫階段、發(fā)送同步期間新寫命令到從庫階段;
建立連接:從庫會和主庫建立連接,從庫執(zhí)行 replicaof 并發(fā)送 psync 命令并告訴主庫即將進行同步,主庫確認回復(fù)后,主從庫間就開始同步了。
主庫同步數(shù)據(jù)給從庫:master 執(zhí)行 bgsave
命令生成 RDB 文件,并將文件發(fā)送給從庫,同時主庫為每一個 slave 開辟一塊 replication buffer 緩沖區(qū)記錄從生成 RDB 文件開始收到的所有寫命令。從庫保存 RDB 并清空數(shù)據(jù)庫再加載 RDB 數(shù)據(jù)到內(nèi)存中。
發(fā)送 RDB 之后接收到的新寫命令到從庫:在生成 RDB 文件之后的寫操作并沒有記錄到剛剛的 RDB 文件中,為了保證主從庫數(shù)據(jù)的一致性,所以主庫會在內(nèi)存中使用一個叫 replication buffer 記錄 RDB 文件生成后的所有寫操作。并將里面的數(shù)據(jù)發(fā)送到 slave。
面試官:主從庫間的網(wǎng)絡(luò)斷了咋辦?斷開后要重新全量復(fù)制么?
在 Redis 2.8 之前,如果主從庫在命令傳播時出現(xiàn)了網(wǎng)絡(luò)閃斷,那么,從庫就會和主庫重新進行一次全量復(fù)制,開銷非常大。
從 Redis 2.8 開始,網(wǎng)絡(luò)斷了之后,主從庫會采用增量復(fù)制的方式繼續(xù)同步。
增量復(fù)制:用于網(wǎng)絡(luò)中斷等情況后的復(fù)制,只將中斷期間主節(jié)點執(zhí)行的寫命令發(fā)送給從節(jié)點,與全量復(fù)制相比更加高效。
斷開重連增量復(fù)制的實現(xiàn)奧秘就是 repl_backlog_buffer
緩沖區(qū),不管在什么時候 master 都會將寫指令操作記錄在 repl_backlog_buffer
中,因為內(nèi)存有限, repl_backlog_buffer
是一個定長的環(huán)形數(shù)組,如果數(shù)組內(nèi)容滿了,就會從頭開始覆蓋前面的內(nèi)容。
master 使用 master_repl_offset
記錄自己寫到的位置偏移量,slave 則使用 slave_repl_offset
記錄已經(jīng)讀取到的偏移量。
當主從斷開重連后,slave 會先發(fā)送 psync 命令給 master,同時將自己的 runID
,slave_repl_offset
發(fā)送給 master。
master 只需要把 master_repl_offset
與 slave_repl_offset
之間的命令同步給從庫即可。
增量復(fù)制執(zhí)行流程如下圖:
面試官:那完成全量同步后,正常運行過程中如何同步數(shù)據(jù)呢?
當主從庫完成了全量復(fù)制,它們之間就會一直維護一個網(wǎng)絡(luò)連接,主庫會通過這個連接將后續(xù)陸續(xù)收到的命令操作再同步給從庫,這個過程也稱為基于長連接的命令傳播,使用長連接的目的就是避免頻繁建立連接導(dǎo)致的開銷。
面試官:可以呀,知道這么多,你知道 哨兵集群原理么?
哨兵是 Redis 的一種運行模式,它專注于對 Redis 實例(主節(jié)點、從節(jié)點)運行狀態(tài)的監(jiān)控,并能夠在主節(jié)點發(fā)生故障時通過一系列的機制實現(xiàn)選主及主從切換,實現(xiàn)故障轉(zhuǎn)移,確保整個 Redis 系統(tǒng)的可用性。
他的架構(gòu)圖如下:
Redis 哨兵具備的能力有如下幾個:
監(jiān)控:持續(xù)監(jiān)控 master 、slave 是否處于預(yù)期工作狀態(tài)。
自動切換主庫:當 Master 運行故障,哨兵啟動自動故障恢復(fù)流程:從 slave 中選擇一臺作為新 master。
通知:讓 slave 執(zhí)行 replicaof ,與新的 master 同步;并且通知客戶端與新 master 建立連接。
面試官:哨兵之間是如何知道彼此的?
哨兵與 master 建立通信,利用 master 提供發(fā)布/訂閱機制發(fā)布自己的信息,比如身高體重、是否單身、IP、端口……
master 有一個 __sentinel__:hello
的專用通道,用于哨兵之間發(fā)布和訂閱消息。這就好比是 __sentinel__:hello
微信群,哨兵利用 master 建立的微信群發(fā)布自己的消息,同時關(guān)注其他哨兵發(fā)布的消息。
面試官:哨兵之間雖然建立連接了,但是還需要和 slave 建立連接,不然沒法監(jiān)控他們呀,如何知道 slave 并監(jiān)控他們的?
關(guān)鍵還是利用 master 來實現(xiàn),哨兵向 master 發(fā)送 INFO
命令, master 掌門自然是知道自己門下所有的 salve 小弟的。所以 master 接收到命令后,便將 slave 列表告訴哨兵。
哨兵根據(jù) master 響應(yīng)的 slave 名單信息與每一個 salve 建立連接,并且根據(jù)這個連接持續(xù)監(jiān)控哨兵。
面試官:除了哨兵以外,還有其他的高可用手段么?
有 Cluster 集群實現(xiàn)高可用,哨兵集群監(jiān)控的 Redis 集群是主從架構(gòu),無法很想拓展。使用 Redis Cluster 集群,主要解決了大數(shù)據(jù)量存儲導(dǎo)致的各種慢問題,同時也便于橫向拓展。
在面向百萬、千萬級別的用戶規(guī)模時,橫向擴展的 Redis 切片集群會是一個非常好的選擇。
面試官:什么是 Cluster 集群?
Redis 集群是一種分布式數(shù)據(jù)庫方案,集群通過分片(sharding)來進行數(shù)據(jù)管理(「分治思想」的一種實踐),并提供復(fù)制和故障轉(zhuǎn)移功能。
將數(shù)據(jù)劃分為 16384 的 slots,每個節(jié)點負責一部分槽位。槽位的信息存儲于每個節(jié)點中。
它是去中心化的,如圖所示,該集群有三個 Redis 節(jié)點組成,每個節(jié)點負責整個集群的一部分數(shù)據(jù),每個節(jié)點負責的數(shù)據(jù)多少可能不一樣。
三個節(jié)點相互連接組成一個對等的集群,它們之間通過 Gossip
協(xié)議相互交互集群信息,最后每個節(jié)點都保存著其他節(jié)點的 slots 分配情況。
面試官:哈希槽又是如何映射到 Redis 實例上呢?
根據(jù)鍵值對的 key,使用 CRC16 算法,計算出一個 16 bit 的值;
將 16 bit 的值對 16384 執(zhí)行取模,得到 0 ~ 16383 的數(shù)表示 key 對應(yīng)的哈希槽。
根據(jù)該槽信息定位到對應(yīng)的實例。
鍵值對數(shù)據(jù)、哈希槽、Redis 實例之間的映射關(guān)系如下:
面試官:Cluster 如何實現(xiàn)故障轉(zhuǎn)移?
Redis 集群節(jié)點采用 Gossip
協(xié)議來廣播自己的狀態(tài)以及自己對整個集群認知的改變。比如一個節(jié)點發(fā)現(xiàn)某個節(jié)點失聯(lián)了 (PFail),它會將這條信息向整個集群廣播,其它節(jié)點也就可以收到這點失聯(lián)信息。
如果一個節(jié)點收到了某個節(jié)點失聯(lián)的數(shù)量 (PFail Count) 已經(jīng)達到了集群的大多數(shù),就可以標記該節(jié)點為確定下線狀態(tài) (Fail),然后向整個集群廣播,強迫其它節(jié)點也接收該節(jié)點已經(jīng)下線的事實,并立即對該失聯(lián)節(jié)點進行主從切換。
面試官:客戶端又怎么確定訪問的數(shù)據(jù)到底分布在哪個實例上呢?
Redis 實例會將自己的哈希槽信息通過 Gossip 協(xié)議發(fā)送給集群中其他的實例,實現(xiàn)了哈希槽分配信息的擴散。
這樣,集群中的每個實例都有所有哈希槽與實例之間的映射關(guān)系信息。
當客戶端連接任何一個實例,實例就將哈希槽與實例的映射關(guān)系響應(yīng)給客戶端,客戶端就會將哈希槽與實例映射信息緩存在本地。
當客戶端請求時,會計算出鍵所對應(yīng)的哈希槽,再通過本地緩存的哈希槽實例映射信息定位到數(shù)據(jù)所在實例上,再將請求發(fā)送給對應(yīng)的實例。
面試官:什么是 Redis 重定向機制?
哈希槽與實例之間的映射關(guān)系由于新增實例或者負載均衡重新分配導(dǎo)致改變了,客戶端將請求發(fā)送到實例上,這個實例沒有相應(yīng)的數(shù)據(jù),該 Redis 實例會告訴客戶端將請求發(fā)送到其他的實例上。
Redis 通過 MOVED 錯誤和 ASK 錯誤告訴客戶端。
MOVED錯誤(負載均衡,數(shù)據(jù)已經(jīng)遷移到其他實例上):當客戶端將一個鍵值對操作請求發(fā)送給某個實例,而這個鍵所在的槽并非由自己負責的時候,該實例會返回一個 MOVED 錯誤指引轉(zhuǎn)向正在負責該槽的節(jié)點。
同時,客戶端還會更新本地緩存,將該 slot 與 Redis 實例對應(yīng)關(guān)系更新正確。
如果某個 slot 的數(shù)據(jù)比較多,部分遷移到新實例,還有一部分沒有遷移。
如果請求的 key 在當前節(jié)點找到就直接執(zhí)行命令,否則時候就需要 ASK 錯誤響應(yīng)了。
槽部分遷移未完成的情況下,如果需要訪問的 key 所在 Slot 正在從 實例 1 遷移到 實例 2(如果 key 已經(jīng)不在實例 1),實例 1 會返回客戶端一條 ASK 報錯信息:客戶端請求的 key 所在的哈希槽正在遷移到實例 2 上,你先給實例 2 發(fā)送一個 ASKING 命令,接著發(fā)發(fā)送操作命令。
比如客戶端請求定位到 key = 「公眾號:碼哥字節(jié)」的槽 16330 在實例 172.17.18.1 上,節(jié)點 1 如果找得到就直接執(zhí)行命令,否則響應(yīng) ASK 錯誤信息,并指引客戶端轉(zhuǎn)向正在遷移的目標節(jié)點 172.17.18.2。
注意:ASK 錯誤指令并不會更新客戶端緩存的哈希槽分配信息。
關(guān)于“Redis中有哪些高頻面試題”這篇文章就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,使各位可以學(xué)到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。