這 40 道 Redis 面試題讓你不再慌(附答案)

爲了做好大家面試路上的助攻手,對於 Redis 這塊心裏還沒底的同學,特整理 40 道Redis常見面試題,讓你面試不慌,爭取 Offer 拿到手軟!

1、什麼是 Redis?
Redis 是完全開源免費的,遵守 BSD 協議,是一個高性能的 key-value 數據庫。

Redis 與其他 key - value 緩存產品相比有以下三個特點:

Redis 支持數據的持久化,可以將內存中的數據保存在磁盤中,重啓的時候可以再次加載進行使用。

Redis 不僅僅支持簡單的 key-value 類型的數據,同時還提供 list,set,zset,hash 等數據結構的存儲。

Redis 支持數據的備份,即 master-slave 模式的數據備份。

Redis 優勢:

性能極高:Redis 能讀的速度是 110000 次/s,寫的速度是 81000 次/s。

豐富的數據類型:Redis 支持二進制案例的 Strings,Lists,Hashes,Sets 及 Ordered Sets 數據類型操作。

原子:Redis 的所有操作都是原子性的,意思就是要麼成功執行要麼失敗完全不執行。單個操作是原子性的。多個操作也支持事務,即原子性,通過 MULTI 和 EXEC 指令包起來。

豐富的特性:Redis 還支持 publish/subscribe,通知,key 過期等等特性。

Redis 與其他 key-value 存儲有什麼不同?

Redis 有着更爲複雜的數據結構並且提供對他們的原子性操作,這是一個不同於其他數據庫的進化路徑。Redis 的數據類型都是基於基本數據結構的同時對程序員透明,無需進行額外的抽象。

Redis 運行在內存中但是可以持久化到磁盤,所以在對不同數據集進行高速讀寫時需要權衡內存,因爲數據量不能大於硬件內存。在內存數據庫方面的另一個優點是,相比在磁盤上相同的複雜的數據結構,在內存中操作起來非常簡單,這樣 Redis 可以做很多內部複雜性很強的事情。同時,在磁盤格式方面他們是緊湊的以追加的方式產生的,因爲他們並不需要進行隨機訪問。

2、Redis 的數據類型?
Redis 支持五種數據類型:string(字符串),hash(哈希),list(列表),set(集合)及 zsetsorted set(有序集合)。

我們實際項目中比較常用的是 string,hash 如果你是 Redis 中高級用戶,還需要加上下面幾種數據結構 HyperLogLog、Geo、Pub/Sub。

如果你說還玩過 Redis Module,像 BloomFilter,RedisSearch,Redis-ML,面試官的眼睛就開始發亮了。

3、使用 Redis 有哪些好處?
速度快,因爲數據存在內存中,類似於 HashMap,HashMap 的優勢就是查找和操作的時間複雜度都是 O1)

支持豐富數據類型,支持 string,list,set,Zset,hash 等

支持事務,操作都是原子性,所謂的原子性就是對數據的更改要麼全部執行,要麼全部不執行

豐富的特性,可用於緩存,消息,按 key 設置過期時間,過期後將會自動刪除

4、Redis 相比 Memcached 有哪些優勢?
Memcached 所有的值均是簡單的字符串,Redis 作爲其替代者,支持更爲豐富的數據類

Redis 的速度比 Memcached 快很多

Redis 可以持久化其數據

5、Memcache 與 Redis 的區別都有哪些?
存儲方式 Memecache 把數據全部存在內存之中,斷電後會掛掉,數據不能超過內存大小。Redis 有部分存在硬盤上,這樣能保證數據的持久性。

數據支持類型 Memcache 對數據類型支持相對簡單。Redis 有複雜的數據類型。

使用底層模型不同 它們之間底層實現方式 以及與客戶端之間通信的應用協議不一樣。Redis 直接自己構建了 VM 機制 ,因爲一般的系統調用系統函數的話,會浪費一定的時間去移動和請求。

6、Redis 是單進程單線程的?
Redis 是單進程單線程的,Redis 利用隊列技術將併發訪問變爲串行訪問,消除了傳統數據庫串行控制的開銷。

7、一個字符串類型的智能存儲最大容量是多少?
512M。

8、Redis 的持久化機制是什麼?各自的優缺點?
Redis提供兩種持久化機制 RDB 和 AOF 機制:

RDB(Redis DataBase)持久化方式:是指用數據集快照的方式半持久化模式記錄 Redis 數據庫的所有鍵值對,在某個時間點將數據寫入一個臨時文件,持久化結束後,用這個臨時文件替換上次持久化的文件,達到數據恢復。

優點:

只有一個文件 dump.rdb,方便持久化。

容災性好,一個文件可以保存到安全的磁盤。

性能最大化,fork 子進程來完成寫操作,讓主進程繼續處理命令,所以是 IO 最大化。使用單獨子進程來進行持久化,主進程不會進行任何 IO 操作,保證了 Redis的高性能。

相對於數據集大時,比 AOF 的啓動效率更高。

缺點:數據安全性低。RDB 是間隔一段時間進行持久化,如果持久化之間 Redis 發生故障,會發生數據丟失。所以這種方式更適合數據要求不嚴謹的時候AOF(Append-only file)持久化方式:是指所有的命令行記錄以 Redis 命令請求協議的格式完全持久化存儲保存爲 aof 文件。

優點:

數據安全,aof 持久化可以配置 appendfsync 屬性,有 always,每進行一次命令操作就記錄到 aof 文件中一次。

通過 append 模式寫文件,即使中途服務器宕機,可以通過 redis-check-aof 工具解決數據一致性問題。

AOF 機制的 rewrite 模式。AOF 文件沒被 rewrite 之前(文件過大時會對命令進行合併重寫),可以刪除其中的某些命令(比如誤操作的 flushall)

缺點:

AOF 文件比 RDB 文件大,且恢復速度慢。

數據集大的時候,比 RDB 啓動效率低。

9、Redis 常見性能問題和解決方案
Master 最好不要寫內存快照,如果 Master 寫內存快照,save 命令調度 rdbSave函數,會阻塞主線程的工作,當快照比較大時對性能影響是非常大的,會間斷性暫停服務。

如果數據比較重要,某個 Slave 開啓 AOF 備份數據,策略設置爲每秒同步一。

爲了主從複製的速度和連接的穩定性,Master 和 Slave 最好在同一個局域網。

儘量避免在壓力很大的主庫上增加從。

主從複製不要用圖狀結構,用單向鏈表結構更爲穩定,即:Master <- Slave1<- Slave2 <- Slave3……這樣的結構方便解決單點故障問題,實現 Slave 對 Master 的替換。如果 Master 掛了,可以立刻啓用 Slave1 做 Master,其他不變。

10、Redis 過期鍵的刪除策略?
定時刪除:在設置鍵的過期時間的同時,創建一個定時器 timer。讓定時器在鍵的過期時間來臨時,立即執行對鍵的刪除操作。

惰性刪除:放任鍵過期不管,但是每次從鍵空間中獲取鍵時,都檢查取得的鍵是否過期,如果過期的話,就刪除該鍵;如果沒有過期,就返回該鍵。

定期刪除:每隔一段時間程序就對數據庫進行一次檢查,刪除裏面的過期鍵。至於要刪除多少過期鍵,以及要檢查多少個數據庫,則由算法決定。

11、Redis 的回收策略(淘汰策略)?
volatile-lru:從已設置過期時間的數據集(server.db[i].expires)中挑選最近最少使用的數據淘汰

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

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

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

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

no-enviction(驅逐):禁止驅逐數據

注意這裏的 6 種機制,volatile 和 allkeys 規定了是對已設置過期時間的數據集淘汰數據還是從全部數據集淘汰數據,後面的 lru、ttl 以及 random 是三種不同的淘汰策略,再加上一種 no-enviction 永不回收的策略。

使用策略規則:

如果數據呈現冪律分佈,也就是一部分數據訪問頻率高,一部分數據訪問頻率低,則使用 allkeys-lru

如果數據呈現平等分佈,也就是所有的數據訪問頻率都相同,則使用 allkeys-random

12、爲什麼 Redis 需要把所有數據放到內存中?
Redis 爲了達到最快的讀寫速度將數據都讀到內存中,並通過異步的方式將數據寫入磁盤。所以 Redis 具有快速和數據持久化的特徵。如果不將數據放在內存中,磁盤 I/O 速度爲嚴重影響 Redis 的性能。在內存越來越便宜的今天,Redis 將會越來越受歡迎。如果設置了最大使用的內存,則數據已有記錄數達到內存限值後不能繼續插入新值。

13、Redis 的同步機制瞭解麼?
Redis 可以使用主從同步,從從同步。第一次同步時,主節點做一次 bgsave,並同時將後續修改操作記錄到內存 buffer,待完成後將 rdb 文件全量同步到複製節點,複製節點接收完成後將 rdb 鏡像加載到內存。加載完成後,再通知主節點將期間修改的操作記錄同步到複製節點進行重放就完成了同步過程。

14、Pipeline 有什麼好處,爲什麼要用 Pipeline?
可以將多次 IO 往返的時間縮減爲一次,前提是 Pipeline 執行的指令之間沒有因果相關性。使用 redis-benchmark 進行壓測的時候可以發現影響 Redis 的 QPS 峯值的一個重要因素是 Pipeline 批次指令的數目。

15、是否使用過 Redis 集羣,集羣的原理是什麼?
Redis Sentinal 着眼於高可用,在 Master 宕機時會自動將 slave 提升爲master,繼續提供服務。

Redis Cluster 着眼於擴展性,在單個 Redis 內存不足時,使用 Cluster 進行分片存儲。

16、Redis 集羣方案什麼情況下會導致整個集羣不可用?
有 A,B,C 三個節點的集羣,在沒有複製模型的情況下,如果節點 B 失敗了,那麼整個集羣就會以爲缺少 5501-11000 這個範圍的槽而不可用。

17、Redis 支持的 Java 客戶端都有哪些?官方推薦用哪個?
Redisson、Jedis、lettuce 等等,官方推薦使用 Redisson。

18、Jedis 與 Redisson 對比有什麼優缺點?
Jedis 是 Redis 的 Java 實現的客戶端,其 API 提供了比較全面的 Redis 命令的支持;Redisson 實現了分佈式和可擴展的 Java 數據結構,和 Jedis 相比,功能較爲簡單,不支持字符串操作,不支持排序、事務、管道、分區等 Redis 特性。

Redisson 的宗旨是促進使用者對 Redis 的關注分離,從而讓使用者能夠將精力更集中地放在處理業務邏輯上。

19、Redis 如何設置密碼及驗證密碼?
設置密碼:config set requirepass 123456

授權密碼:auth 123456

20、說說 Redis 哈希槽的概念?
Redis 集羣沒有使用一致性 hash,而是引入了哈希槽的概念,Redis 集羣有 16384 個哈希槽,每個 key 通過 CRC16 校驗後對 16384 取模來決定放置哪個槽,集羣的每個節點負責一部分 hash 槽。

21、Redis 集羣的主從複製模型是怎樣的?
爲了使在部分節點失敗或者大部分節點無法通信的情況下集羣仍然可用,所以集羣使用了主從複製模型,每個節點都會有 N-1 個複製品。

22、Redis 集羣會有寫操作丟失嗎?爲什麼?
Redis 並不能保證數據的強一致性,這意味着在實際中集羣在特定的條件下可能會丟失寫操作。

23、Redis 集羣之間是如何複製的?
異步複製。

24、Redis 集羣最大節點個數是多少?
16384 個。

25、Redis 集羣如何選擇數據庫?
Redis 集羣目前無法做數據庫選擇,默認在 0 數據庫。

26、怎麼測試 Redis 的連通性?
使用 ping 命令。

27、怎麼理解 Redis 事務?
事務是一個單獨的隔離操作:事務中的所有命令都會序列化、按順序地執行。事務在執行的過程中,不會被其他客戶端發送來的命令請求所打斷。

事務是一個原子操作:事務中的命令要麼全部被執行,要麼全部都不執行。

28、Redis 事務相關的命令有哪幾個?
MULTI、EXEC、DISCARD、WATCH。

29、Redis key 的過期時間和永久有效分別怎麼設置?
EXPIRE 和 PERSIST 命令。

30、Redis 如何做內存優化?
儘可能使用散列表(hashes),散列表(是說散列表裏面存儲的數少)使用的內存非常小,所以你應該儘可能的將你的數據模型抽象到一個散列表裏面。比如你的 Web 系統中有一個用戶對象,不要爲這個用戶的名稱,姓氏,郵箱,密碼設置單獨的 key,而是應該把這個用戶的所有信息存儲到一張散列表裏面。

31、Redis 回收進程如何工作的?
一個客戶端運行了新的命令,添加了新的數據。Redis 檢查內存使用情況,如果大於 maxmemory 的限制,則根據設定好的策略進行回收。一個新的命令被執行,等等。所以我們不斷地穿越內存限制的邊界,通過不斷達到邊界然後不斷地回收回到邊界以下。如果一個命令的結果導致大量內存被使用(例如很大的集合的交集保存到一個新的鍵),不用多久內存限制就會被這個內存使用量超越。

32、都有哪些辦法可以降低 Redis 的內存使用情況呢?
如果你使用的是 32 位的 Redis 實例,可以好好利用 Hash,list,sorted set,set等集合類型數據,因爲通常情況下很多小的 Key-Value 可以用更緊湊的方式存放到一起。

33、Redis 的內存用完了會發生什麼?
如果達到設置的上限,Redis 的寫命令會返回錯誤信息(但是讀命令還可以正常返回。)或者你可以將 Redis 當緩存來使用配置淘汰機制,當 Redis 達到內存上限時會沖刷掉舊的內容。

34、一個 Redis 實例最多能存放多少的 keys?List、Set、Sorted Set 他們最多能存放多少元素?
理論上 Redis 可以處理多達 232 的 keys,並且在實際中進行了測試,每個實例至少存放了 2 億 5 千萬的 keys。我們正在測試一些較大的值。任何 list、set、和 sorted set 都可以放 232 個元素。換句話說,Redis 的存儲極限是系統中的可用內存值。

35、MySQL 裏有 2000w 數據,Redis 中只存 20w 的數據,如何保證 Redis 中的數據都是熱點數據?
Redis 內存數據集大小上升到一定大小的時候,就會施行數據淘汰策略。

相關知識:Redis 提供 6 種數據淘汰策略:

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

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

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

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

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

no-enviction(驅逐):禁止驅逐數據

36、Redis 最適合的場景?
會話緩存(Session Cache),最常用的一種使用 Redis 的情景是會話緩存(session cache)。用 Redis 緩存會話比其他存儲(如 Memcached)的優勢在於:Redis 提供持久化。當維護一個不是嚴格要求一致性的緩存時,如果用戶的購物車信息全部丟失,大部分人都會不高興的,現在,他們還會這樣嗎?幸運的是,隨着 Redis 這些年的改進,很容易找到怎麼恰當的使用 Redis 來緩存會話的文檔。甚至廣爲人知的商業平臺Magento 也提供 Redis 的插件。

全頁緩存(FPC),除基本的會話 token 之外,Redis 還提供很簡便的 FPC 平臺。回到一致性問題,即使重啓了 Redis 實例,因爲有磁盤的持久化,用戶也不會看到頁面加載速度的下降,這是一個極大改進,類似 PHP 本地 FPC。再次以 Magento 爲例,Magento提供一個插件來使用 Redis 作爲全頁緩存後端。此外,對 WordPress 的用戶來說,Pantheon 有一個非常好的插件 wp-redis,這個插件能幫助你以最快速度加載你曾瀏覽過的頁面。

隊列,Reids 在內存存儲引擎領域的一大優點是提供 list 和 set 操作,這使得 Redis能作爲一個很好的消息隊列平臺來使用。Redis 作爲隊列使用的操作,就類似於本地程序語言(如 Python)對 list 的 push/pop 操作。如果你快速地在 Google中搜索“Redis queues”,你馬上就能找到大量的開源項目,這些項目的目的就是利用 Redis 創建非常好的後端工具,以滿足各種隊列需求。例如,Celery 有一個後臺就是使用 Redis 作爲 broker,你可以從這裏去查看。

排行榜/計數器,Redis 在內存中對數字進行遞增或遞減的操作實現的非常好。集合(Set)和有序集合(Sorted Set)也使得我們在執行這些操作的時候變的非常簡單,Redis 只是正好提供了這兩種數據結構。所以,我們要從排序集合中獲取到排名最靠前的 10 個用戶——我們稱之爲“user_scores”,我們只需要像下面一樣執行即可:當然,這是假定你是根據你用戶的分數做遞增的排序。如果你想返回用戶及用戶的分數,你需要這樣執行:ZRANGE user_scores 0 10 WITHSCORES Agora Games 就是一個很好的例子,用 Ruby 實現的,它的排行榜就是使用 Redis 來存儲數據的,你可以在這裏看到。

發佈/訂閱,最後(但肯定不是最不重要的)是 Redis 的發佈/訂閱功能。發佈/訂閱的使用場景確實非常多。我已看見人們在社交網絡連接中使用,還可作爲基於發佈/訂閱的腳本觸發器,甚至用 Redis 的發佈/訂閱功能來建立聊天系統!

37、假如 Redis 裏面有 1 億個 key,其中有 10w 個 key 是以某個固定的已知的前綴開頭的,如果將它們全部找出來?
使用 keys 指令可以掃出指定模式的 key 列表。

對方接着追問:如果這個 Redis 正在給線上的業務提供服務,那使用 keys 指令會有什麼問題?

這個時候你要回答 Redis 關鍵的一個特性:Redis 的單線程的。keys 指令會導致線程阻塞一段時間,線上服務會停頓,直到指令執行完畢,服務才能恢復。這個時候可以使用 scan 指令,scan 指令可以無阻塞地提取出指定模式的 key 列表,但是會有一定的重複概率,在客戶端做一次去重就可以了,但是整體所花費的時間會比直接用 keys 指令長。

38、如果有大量的 key 需要設置同一時間過期,一般需要注意什麼?
如果大量的 key 過期時間設置的過於集中,到過期的那個時間點,Redis 可能會出現短暫的卡頓現象。一般需要在時間上加一個隨機值,使得過期時間分散一些。

39、使用過 Redis 做異步隊列麼,你是怎麼用的?
答:一般使用 list 結構作爲隊列,rpush 生產消息,lpop 消費消息。當 lpop 沒有消息的時候,要適當 sleep 一會再重試。如果對方追問可不可以不用 sleep 呢?list 還有個指令叫 blpop,在沒有消息的時候,它會阻塞住直到消息到來。如果對方追問能不能生產一次消費多次呢?使用 pub/sub 主題訂閱者模式,可以實現1:N 的消息隊列。

如果對方追問 pub/sub 有什麼缺點?

在消費者下線的情況下,生產的消息會丟失,得使用專業的消息隊列如 RabbitMQ等。

如果對方追問 Redis 如何實現延時隊列?

我估計現在你很想把面試官一棒打死如果你手上有一根棒球棍的話,怎麼問得這麼詳細。但是你很剋制,然後神態自若地回答道:使用 sortedset,拿時間戳作爲score,消息內容作爲 key 調用 zadd 來生產消息,消費者用 zrangebyscore 指令獲取 N 秒之前的數據輪詢進行處理。

到這裏,面試官暗地裏已經對你豎起了大拇指。但是他不知道的是此刻你卻豎起了中指,在椅子背後。

40、使用過 Redis 分佈式鎖麼,它是什麼回事?
先拿 setnx 來爭搶鎖,搶到之後,再用 expire 給鎖加一個過期時間防止鎖忘記了釋放。

這時候對方會告訴你說你回答得不錯,然後接着問如果在 setnx 之後執行 expire之前進程意外 crash 或者要重啓維護了,那會怎麼樣?這時候你要給予驚訝的反饋:唉,是喔,這個鎖就永遠得不到釋放了。緊接着你需要抓一抓自己的腦袋,故作思考片刻,好像接下來的結果是你主動思考出來的,然後回答:我記得 set 指令有非常複雜的參數,這個應該是可以同時把 setnx 和expire 合成一條指令來用的!

對方這時會顯露笑容,心裏開始默唸:摁,這小子還不錯。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章