1蛛勉、什么是 Redis?
Redis 是完全開源免費(fèi)的,遵守 BSD 協(xié)議毡熏,是一個高性能的 key-value 數(shù)據(jù)庫侣诵。
Redis 與其他 key - value 緩存產(chǎn)品有以下三個特點(diǎn):
(1)Redis 支持?jǐn)?shù)據(jù)的持久化,可以將內(nèi)存中的數(shù)據(jù)保存在磁盤中财搁,重啟的時候可以再次加載進(jìn)行使用尖奔。
(2)Redis 不僅僅支持簡單的 key-value 類型的數(shù)據(jù),同時還提供 list越锈,set甘凭,zset,hash 等數(shù)據(jù)結(jié)構(gòu)的存儲德撬。
(3)Redis 支持?jǐn)?shù)據(jù)的備份躲胳,即 master-slave 模式的數(shù)據(jù)備份。
Redis 優(yōu)勢
(1)性能極高 – Redis 能讀的速度是 110000 次/s,寫的速度是 81000 次/s 隆檀。
(2)豐富的數(shù)據(jù)類型 – Redis 支持二進(jìn)制案例的 Strings, Lists, Hashes, Sets 及Ordered Sets 數(shù)據(jù)類型操作粹湃。
(3)原子 – Redis 的所有操作都是原子性的,意思就是要么成功執(zhí)行要么失敗完全不執(zhí)行裳仆。單個操作是原子性的孤钦。多個操作也支持事務(wù),即原子性静袖。
(4)豐富的特性 – Redis 還支持 publish/subscribe, 通知, key 過期等等特性壳猜。
Redis 與其他 key-value 存儲有什么不同?
(1)Redis 有著更為復(fù)雜的數(shù)據(jù)結(jié)構(gòu)并且提供對他們的原子性操作喘帚,這是一個不同于其他數(shù)據(jù)庫的進(jìn)化路徑。Redis 的數(shù)據(jù)類型都是基于基本數(shù)據(jù)結(jié)構(gòu)的同時對程序員透明若未,無需進(jìn)行額外的抽象倾鲫。
(2)Redis 運(yùn)行在內(nèi)存中但是可以持久化到磁盤乌昔,所以在對不同數(shù)據(jù)集進(jìn)行高速讀寫時需要權(quán)衡內(nèi)存,因?yàn)閿?shù)據(jù)量不能大于硬件內(nèi)存供屉。在內(nèi)存數(shù)據(jù)庫方面的另一個優(yōu)點(diǎn)是溺蕉,相比在磁盤上相同的復(fù)雜的數(shù)據(jù)結(jié)構(gòu),在內(nèi)存中操作起來非常簡單哗魂,這樣 Redis可以做很多內(nèi)部復(fù)雜性很強(qiáng)的事情漓雅。同時故硅,在磁盤格式方面他們是緊湊的以追加的方式產(chǎn)生的,因?yàn)樗麄儾⒉恍枰M(jìn)行隨機(jī)訪問。
2腾誉、Redis 的數(shù)據(jù)類型利职?
答:Redis 支持五種數(shù)據(jù)類型:string(字符串),hash(哈希)跷敬,list(列表)热押,set(集合)及 zsetsorted set:有序集合)斤寇。
我們實(shí)際項(xiàng)目中比較常用的是 string娘锁,hash 如果你是 Redis 中高級用戶饺鹃,還需要加上下面幾種數(shù)據(jù)結(jié)構(gòu) HyperLogLog、Geo镊屎、Pub/Sub茄螃。
如果你說還玩過 Redis Module责蝠,像 BloomFilter,RedisSearch齿拂,Redis-ML肴敛,面試官得眼睛就開始發(fā)亮了医男。
3、使用 Redis 有哪些好處刀森?
(1)速度快报账,因?yàn)閿?shù)據(jù)存在內(nèi)存中,類似于 HashMap榜晦,HashMap 的優(yōu)勢就是查找和操作的時間復(fù)雜度都是 O1)
(2)支持豐富數(shù)據(jù)類型羽圃,支持 string,list识窿,set腕扶,Zset,hash 等
(3)支持事務(wù)脓恕,操作都是原子性窿侈,所謂的原子性就是對數(shù)據(jù)的更改要么全部執(zhí)行,要么全部不執(zhí)行
(4)豐富的特性:可用于緩存乃秀,消息跺讯,按 key 設(shè)置過期時間殉农,過期后將會自動刪除
4、Redis 相比 Memcached 有哪些優(yōu)勢愈污?
(1)Memcached 所有的值均是簡單的字符串轮傍,redis 作為其替代者创夜,支持更為豐富的數(shù)據(jù)類
(2)Redis 的速度比 Memcached 快很
(3)Redis 可以持久化其數(shù)據(jù)
5、Memcache 與 Redis 的區(qū)別都有哪些揍魂?
(1)存儲方式 Memecache 把數(shù)據(jù)全部存在內(nèi)存之中棚瘟,斷電后會掛掉偎蘸,數(shù)據(jù)不能超過內(nèi)存大小。 Redis 有部份存在硬盤上限书,這樣能保證數(shù)據(jù)的持久性章咧。
(2)數(shù)據(jù)支持類型 Memcache 對數(shù)據(jù)類型支持相對簡單赁严。 Redis 有復(fù)雜的數(shù)據(jù)類型。
(3)使用底層模型不同 它們之間底層實(shí)現(xiàn)方式 以及與客戶端之間通信的應(yīng)用協(xié)議不一樣卤档。 Redis 直接自己構(gòu)建了 VM 機(jī)制 程剥,因?yàn)橐话愕南到y(tǒng)調(diào)用系統(tǒng)函數(shù)的話织鲸,會浪費(fèi)一定的時間去移動和請求。
6稳诚、Redis 是單進(jìn)程單線程的盾饮?
答:Redis 是單進(jìn)程單線程的丘损,redis 利用隊(duì)列技術(shù)將并發(fā)訪問變?yōu)榇性L問,消除了傳統(tǒng)數(shù)據(jù)庫串行控制的開銷衔蹲。
7呈础、一個字符串類型的值能存儲最大容量是多少而钞?
答:512M
8、Redis 的持久化機(jī)制是什么撬陵?各自的優(yōu)缺點(diǎn)?
Redis提供兩種持久化機(jī)制 RDB 和 AOF 機(jī)制:
1蟋定、RDBRedis DataBase)持久化方式:
是指用數(shù)據(jù)集快照的方式半持久化模式)記錄 redis 數(shù)據(jù)庫的所有鍵值對,在某個時間點(diǎn)將數(shù)據(jù)寫入一個臨時文件驶兜,持久化結(jié)束后远寸,用這個臨時文件替換上次持久化的文件,達(dá)到數(shù)據(jù)恢復(fù)蝇狼。
優(yōu)點(diǎn):
(1)只有一個文件 dump.rdb迅耘,方便持久化监署。
(2)容災(zāi)性好,一個文件可以保存到安全的磁盤栖秕。
(3)性能最大化簇捍,fork 子進(jìn)程來完成寫操作俏拱,讓主進(jìn)程繼續(xù)處理命令,所以是 IO最大化事格。使用單獨(dú)子進(jìn)程來進(jìn)行持久化驹愚,主進(jìn)程不會進(jìn)行任何 IO 操作劣纲,保證了 redis的高性能)
(4)相對于數(shù)據(jù)集大時,比 AOF 的啟動效率更高蒸甜。
缺點(diǎn):
數(shù)據(jù)安全性低柠新。RDB 是間隔一段時間進(jìn)行持久化辉巡,如果持久化之間 redis 發(fā)生故障郊楣,會發(fā)生數(shù)據(jù)丟失。所以這種方式更適合數(shù)據(jù)要求不嚴(yán)謹(jǐn)?shù)臅r候
2钥组、AOFAppend-only file)持久化方式:
是指所有的命令行記錄以 redis 命令請求協(xié)議的格式完全持久化存儲)保存為 aof 文件今瀑。
優(yōu)點(diǎn):
(1)數(shù)據(jù)安全橘荠,aof 持久化可以配置 appendfsync 屬性,有 always挺份,每進(jìn)行一次命令操作就記錄到 aof 文件中一次贮懈。
(2)通過 append 模式寫文件朵你,即使中途服務(wù)器宕機(jī),可以通過 redis-check-aof工具解決數(shù)據(jù)一致性問題伦吠。
(3)AOF 機(jī)制的 rewrite 模式毛仪。AOF 文件沒被 rewrite 之前(文件過大時會對命令進(jìn)行合并重寫)芯勘,可以刪除其中的某些命令(比如誤操作的 flushall))
缺點(diǎn):
(1)AOF 文件比 RDB 文件大荷愕,且恢復(fù)速度慢棍矛。
(2)數(shù)據(jù)集大的時候抛杨,比 rdb 啟動效率低怖现。
9、Redis 常見性能問題和解決方案:
(1)Master 最好不要寫內(nèi)存快照潘拨,如果 Master 寫內(nèi)存快照铁追,save 命令調(diào)度 rdbSave函數(shù)茫船,會阻塞主線程的工作透硝,當(dāng)快照比較大時對性能影響是非常大的,會間斷性暫停服務(wù)
(2)如果數(shù)據(jù)比較重要埋泵,某個 Slave 開啟 AOF 備份數(shù)據(jù)罪治,策略設(shè)置為每秒同步一次觉义。
(3)為了主從復(fù)制的速度和連接的穩(wěn)定性,Master 和 Slave 最好在同一個局域網(wǎng)
(4)盡量避免在壓力很大的主庫上增加從
(5)主從復(fù)制不要用圖狀結(jié)構(gòu)霉撵,用單向鏈表結(jié)構(gòu)更為穩(wěn)定徒坡,即:Master <- Slave1<- Slave2 <- Slave3…這樣的結(jié)構(gòu)方便解決單點(diǎn)故障問題瘤缩,實(shí)現(xiàn) Slave 對 Master的替換剥啤。如果 Master 掛了,可以立刻啟用 Slave1 做 Master刻诊,其他不變坏逢。
10、redis 過期鍵的刪除策略?
(1)定時刪除:在設(shè)置鍵的過期時間的同時浮入,創(chuàng)建一個定時器 timer). 讓定時器在鍵的過期時間來臨時羊异,立即執(zhí)行對鍵的刪除操作野舶。
(2)惰性刪除:放任鍵過期不管,但是每次從鍵空間中獲取鍵時睹欲,都檢查取得的鍵是否過期窘疮,如果過期的話冀墨,就刪除該鍵;如果沒有過期,就返回該鍵蔚出。
(3)定期刪除:每隔一段時間程序就對數(shù)據(jù)庫進(jìn)行一次檢查骄酗,刪除里面的過期鍵岔乔。至于要刪除多少過期鍵雏门,以及要檢查多少個數(shù)據(jù)庫掸掏,則由算法決定丧凤。
11步脓、Redis 的回收策略(淘汰策略)?
volatile-lru:從已設(shè)置過期時間的數(shù)據(jù)集(server.db[i].expires)中挑選最近最少使用的數(shù)據(jù)淘汰
volatile-ttl:從已設(shè)置過期時間的數(shù)據(jù)集(server.db[i].expires)中挑選將要過期的數(shù)據(jù)淘汰
volatile-random:從已設(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ù)
注意這里的 6 種機(jī)制靴患,volatile 和 allkeys 規(guī)定了是對已設(shè)置過期時間的數(shù)據(jù)集淘汰數(shù)據(jù)還是從全部數(shù)據(jù)集淘汰數(shù)據(jù)鸳君,后面的 lru或颊、ttl 以及 random 是三種不同的淘汰策略,再加上一種 no-enviction 永不回收的策略醉顽。
使用策略規(guī)則:
(1)如果數(shù)據(jù)呈現(xiàn)冪律分布游添,也就是一部分?jǐn)?shù)據(jù)訪問頻率高弹惦,一部分?jǐn)?shù)據(jù)訪問頻率低棠隐,則使用 allkeys-lru
(2)如果數(shù)據(jù)呈現(xiàn)平等分布,也就是所有的數(shù)據(jù)訪問頻率都相同啰扛,則使用allkeys-random
12隐解、為什么 edis 需要把所有數(shù)據(jù)放到內(nèi)存中诫睬?
答 :Redis 為了達(dá)到最快的讀寫速度將數(shù)據(jù)都讀到內(nèi)存中,并通過異步的方式將數(shù)據(jù)寫入磁盤蚓曼。所以 redis 具有快速和數(shù)據(jù)持久化的特征钦扭。如果不將數(shù)據(jù)放在內(nèi)存中客情,磁盤 I/O 速度為嚴(yán)重影響 redis 的性能膀斋。在內(nèi)存越來越便宜的今天,redis 將會越來越受歡迎籽御。如果設(shè)置了最大使用的內(nèi)存惰匙,則數(shù)據(jù)已有記錄數(shù)達(dá)到內(nèi)存限值后不能繼續(xù)插入新值项鬼。
13绘盟、Redis 的同步機(jī)制了解么悯仙?
答:Redis 可以使用主從同步锡垄,從從同步货岭。第一次同步時,主節(jié)點(diǎn)做一次 bgsave屯仗,并同時將后續(xù)修改操作記錄到內(nèi)存 buffer搔谴,待完成后將 rdb 文件全量同步到復(fù)制節(jié)點(diǎn)魁袜,復(fù)制節(jié)點(diǎn)接受完成后將 rdb 鏡像加載到內(nèi)存。加載完成后,再通知主節(jié)點(diǎn)將期間修改的操作記錄同步到復(fù)制節(jié)點(diǎn)進(jìn)行重放就完成了同步過程峰弹。
14店量、Pipeline 有什么好處,為什么要用 pipeline垮卓?
答:可以將多次 IO 往返的時間縮減為一次垫桂,前提是 pipeline 執(zhí)行的指令之間沒有因果相關(guān)性。使用 redis-benchmark 進(jìn)行壓測的時候可以發(fā)現(xiàn)影響 redis 的 QPS峰值的一個重要因素是 pipeline 批次指令的數(shù)目诬滩。
15、是否使用過 Redis 集群灭将,集群的原理是什么疼鸟?
(1)Redis Sentinal 著眼于高可用,在 master 宕機(jī)時會自動將 slave 提升為master庙曙,繼續(xù)提供服務(wù)空镜。
(2)Redis Cluster 著眼于擴(kuò)展性,在單個 redis 內(nèi)存不足時捌朴,使用 Cluster 進(jìn)行分片存儲吴攒。
16、Redis 集群方案什么情況下會導(dǎo)致整個集群不可用砂蔽?
答:有 A洼怔,B,C 三個節(jié)點(diǎn)的集群,在沒有復(fù)制模型的情況下,如果節(jié)點(diǎn) B 失敗了左驾,那么整個集群就會以為缺少 5501-11000 這個范圍的槽而不可用镣隶。
17、Redis 支持的 Java 客戶端都有哪些诡右?官方推薦用哪個安岂?
答:Redisson、Jedis帆吻、lettuce 等等域那,官方推薦使用 Redisson。
18桅锄、Jedis 與 Redisson 對比有什么優(yōu)缺點(diǎn)琉雳?
答:Jedis 是 Redis 的 Java 實(shí)現(xiàn)的客戶端,其 API 提供了比較全面的 Redis 命令的支持友瘤;Redisson 實(shí)現(xiàn)了分布式和可擴(kuò)展的 Java 數(shù)據(jù)結(jié)構(gòu)翠肘,和 Jedis 相比,功能較為簡單辫秧,不支持字符串操作束倍,不支持排序、事務(wù)、管道绪妹、分區(qū)等 Redis 特性甥桂。
Redisson 的宗旨是促進(jìn)使用者對 Redis 的關(guān)注分離,從而讓使用者能夠?qū)⒕Ω械胤旁谔幚順I(yè)務(wù)邏輯上邮旷。
19黄选、Redis 如何設(shè)置密碼及驗(yàn)證密碼?
設(shè)置密碼:config set requirepass 123456
授權(quán)密碼:auth 123456
20婶肩、說說 Redis 哈希槽的概念办陷?
答:Redis 集群沒有使用一致性 hash,而是引入了哈希槽的概念,Redis 集群有16384 個哈希槽律歼,每個 key 通過 CRC16 校驗(yàn)后對 16384 取模來決定放置哪個槽民镜,集群的每個節(jié)點(diǎn)負(fù)責(zé)一部分 hash 槽。
21险毁、Redis 集群的主從復(fù)制模型是怎樣的制圈?
答:為了使在部分節(jié)點(diǎn)失敗或者大部分節(jié)點(diǎn)無法通信的情況下集群仍然可用,所以集群使用了主從復(fù)制模型,每個節(jié)點(diǎn)都會有 N-1 個復(fù)制品.
22畔况、Redis 集群會有寫操作丟失嗎鲸鹦?為什么?
答 :Redis 并不能保證數(shù)據(jù)的強(qiáng)一致性跷跪,這意味這在實(shí)際中集群在特定的條件下可能會丟失寫操作亥鬓。
23、Redis 集群之間是如何復(fù)制的域庇?
答:異步復(fù)制
24、Redis 集群最大節(jié)點(diǎn)個數(shù)是多少覆积?
答:16384 個听皿。
25、Redis 集群如何選擇數(shù)據(jù)庫宽档?
答:Redis 集群目前無法做數(shù)據(jù)庫選擇尉姨,默認(rèn)在 0 數(shù)據(jù)庫。
26吗冤、怎么測試 Redis 的連通性又厉?
答:使用 ping 命令。
27椎瘟、怎么理解 Redis 事務(wù)覆致?
答:
(1)事務(wù)是一個單獨(dú)的隔離操作:事務(wù)中的所有命令都會序列化、按順序地執(zhí)行肺蔚。事務(wù)在執(zhí)行的過程中煌妈,不會被其他客戶端發(fā)送來的命令請求所打斷。
(2)事務(wù)是一個原子操作:事務(wù)中的命令要么全部被執(zhí)行,要么全部都不執(zhí)行璧诵。
28汰蜘、Redis 事務(wù)相關(guān)的命令有哪幾個?
答:MULTI之宿、EXEC族操、DISCARD、WATCH
29比被、Redis key 的過期時間和永久有效分別怎么設(shè)置色难?
答:EXPIRE 和 PERSIST 命令。
30姐赡、Redis 如何做內(nèi)存優(yōu)化莱预?
答:盡可能使用散列表(hashes),散列表(是說散列表里面存儲的數(shù)少)使用的內(nèi)存非常小项滑,所以你應(yīng)該盡可能的將你的數(shù)據(jù)模型抽象到一個散列表里面依沮。比如你的 web 系統(tǒng)中有一個用戶對象,不要為這個用戶的名稱枪狂,姓氏危喉,郵箱,密碼設(shè)置單獨(dú)的 key,而是應(yīng)該把這個用戶的所有信息存儲到一張散列表里面州疾。
31辜限、Redis 回收進(jìn)程如何工作的?
答:一個客戶端運(yùn)行了新的命令严蓖,添加了新的數(shù)據(jù)薄嫡。Redi 檢查內(nèi)存使用情況,如果大于 maxmemory 的限制, 則根據(jù)設(shè)定好的策略進(jìn)行回收颗胡。一個新的命令被執(zhí)行毫深,等等。所以我們不斷地穿越內(nèi)存限制的邊界毒姨,通過不斷達(dá)到邊界然后不斷地回收回到邊界以下哑蔫。如果一個命令的結(jié)果導(dǎo)致大量內(nèi)存被使用(例如很大的集合的交集保存到一個新的鍵),不用多久內(nèi)存限制就會被這個內(nèi)存使用量超越弧呐。
32闸迷、都有哪些辦法可以降低 Redis 的內(nèi)存使用情況呢?
答:如果你使用的是 32 位的 Redis 實(shí)例俘枫,可以好好利用 Hash,list,sorted set,set等集合類型數(shù)據(jù)腥沽,因?yàn)橥ǔG闆r下很多小的 Key-Value 可以用更緊湊的方式存放到一起。
33崩哩、Redis 的內(nèi)存用完了會發(fā)生什么巡球?
答:如果達(dá)到設(shè)置的上限言沐,Redis 的寫命令會返回錯誤信息(但是讀命令還可以正常返回。)或者你可以將 Redis 當(dāng)緩存來使用配置淘汰機(jī)制酣栈,當(dāng) Redis 達(dá)到內(nèi)存上限時會沖刷掉舊的內(nèi)容险胰。
34、一個 Redis 實(shí)例最多能存放多少的 keys矿筝?List起便、Set、Sorted Set 他們最多能存放多少元素窖维?
答:理論上 Redis 可以處理多達(dá) 232 的 keys榆综,并且在實(shí)際中進(jìn)行了測試,每個實(shí)例至少存放了 2 億 5 千萬的 keys铸史。我們正在測試一些較大的值鼻疮。任何 list、set琳轿、和 sorted set 都可以放 232 個元素判沟。換句話說,Redis 的存儲極限是系統(tǒng)中的可用內(nèi)存值崭篡。
35挪哄、MySQL 里有 2000w 數(shù)據(jù),redis 中只存 20w 的數(shù)據(jù)琉闪,如何保證 redis 中的數(shù)據(jù)都是熱點(diǎn)數(shù)據(jù)迹炼?
答:Redis 內(nèi)存數(shù)據(jù)集大小上升到一定大小的時候,就會施行數(shù)據(jù)淘汰策略颠毙。
相關(guān)知識:Redis 提供 6 種數(shù)據(jù)淘汰策略:
volatile-lru:從已設(shè)置過期時間的數(shù)據(jù)集(server.db[i].expires)中挑選最近最少使用的數(shù)據(jù)淘汰
volatile-ttl:從已設(shè)置過期時間的數(shù)據(jù)集(server.db[i].expires)中挑選將要過期的數(shù)據(jù)淘汰
volatile-random:從已設(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ù)
36斯入、Redis 最適合的場景?
1蛀蜜、會話緩存(Session Cache)
最常用的一種使用 Redis 的情景是會話緩存(session cache)咱扣。用 Redis 緩存會話比其他存儲(如 Memcached)的優(yōu)勢在于:Redis 提供持久化。當(dāng)維護(hù)一個不是嚴(yán)格要求一致性的緩存時涵防,如果用戶的購物車信息全部丟失,大部分人都會不高興的沪铭,現(xiàn)在壮池,他們還會這樣嗎? 幸運(yùn)的是杀怠,隨著 Redis 這些年的改進(jìn)椰憋,很容易找到怎么恰當(dāng)?shù)氖褂?Redis 來緩存會話的文檔。甚至廣為人知的商業(yè)平臺Magento 也提供 Redis 的插件赔退。
2橙依、全頁緩存(FPC)
除基本的會話 token 之外证舟,Redis 還提供很簡便的 FPC 平臺〈捌铮回到一致性問題女责,即使重啟了 Redis 實(shí)例,因?yàn)橛写疟P的持久化创译,用戶也不會看到頁面加載速度的下降抵知,這是一個極大改進(jìn),類似 PHP 本地 FPC软族。 再次以 Magento 為例刷喜,Magento提供一個插件來使用 Redis 作為全頁緩存后端。 此外立砸,對 WordPress 的用戶來說掖疮,Pantheon 有一個非常好的插件 wp-redis,這個插件能幫助你以最快速度加載你曾瀏覽過的頁面颗祝。
3浊闪、隊(duì)列
Reids 在內(nèi)存存儲引擎領(lǐng)域的一大優(yōu)點(diǎn)是提供 list 和 set 操作,這使得 Redis能作為一個很好的消息隊(duì)列平臺來使用吐葵。Redis 作為隊(duì)列使用的操作规揪,就類似于本地程序語言(如 Python)對 list 的 push/pop 操作。 如果你快速的在 Google中搜索“Redis queues”温峭,你馬上就能找到大量的開源項(xiàng)目猛铅,這些項(xiàng)目的目的就是利用 Redis 創(chuàng)建非常好的后端工具,以滿足各種隊(duì)列需求凤藏。例如奸忽,Celery 有一個后臺就是使用 Redis 作為 broker,你可以從這里去查看揖庄。
4栗菜,排行榜/計(jì)數(shù)器
Redis 在內(nèi)存中對數(shù)字進(jìn)行遞增或遞減的操作實(shí)現(xiàn)的非常好。集合(Set)和有序集合(Sorted Set)也使得我們在執(zhí)行這些操作的時候變的非常簡單蹄梢,Redis 只是正好提供了這兩種數(shù)據(jù)結(jié)構(gòu)疙筹。所以,我們要從排序集合中獲取到排名最靠前的 10個用戶–我們稱之為“user_scores”禁炒,我們只需要像下面一樣執(zhí)行即可: 當(dāng)然而咆,這是假定你是根據(jù)你用戶的分?jǐn)?shù)做遞增的排序。如果你想返回用戶及用戶的分?jǐn)?shù)幕袱,你需要這樣執(zhí)行: ZRANGE user_scores 0 10 WITHSCORES Agora Games 就是一個很好的例子暴备,用 Ruby 實(shí)現(xiàn)的,它的排行榜就是使用 Redis 來存儲數(shù)據(jù)的们豌,你可以在這里看到涯捻。
5浅妆、發(fā)布/訂閱
最后(但肯定不是最不重要的)是 Redis 的發(fā)布/訂閱功能。發(fā)布/訂閱的使用場景確實(shí)非常多障癌。我已看見人們在社交網(wǎng)絡(luò)連接中使用凌外,還可作為基于發(fā)布/訂閱的腳本觸發(fā)器,甚至用 Redis 的發(fā)布/訂閱功能來建立聊天系統(tǒng)混弥!
37趴乡、假如 Redis 里面有 1 億個 key,其中有 10w 個 key 是以某個固定的已知的前綴開頭的蝗拿,如果將它們?nèi)空页鰜恚?/h1>
答:使用 keys 指令可以掃出指定模式的 key 列表晾捏。
對方接著追問:如果這個 redis 正在給線上的業(yè)務(wù)提供服務(wù),那使用 keys 指令會有什么問題哀托?
這個時候你要回答 redis 關(guān)鍵的一個特性:redis 的單線程的惦辛。keys 指令會導(dǎo)致線程阻塞一段時間,線上服務(wù)會停頓仓手,直到指令執(zhí)行完畢胖齐,服務(wù)才能恢復(fù)。這個時候可以使用 scan 指令嗽冒,scan 指令可以無阻塞的提取出指定模式的 key 列表呀伙,但是會有一定的重復(fù)概率,在客戶端做一次去重就可以了添坊,但是整體所花費(fèi)的時間會比直接用 keys 指令長剿另。
38、如果有大量的 key 需要設(shè)置同一時間過期贬蛙,一般需要注意什么雨女?
答:如果大量的 key 過期時間設(shè)置的過于集中,到過期的那個時間點(diǎn)阳准,redis 可能會出現(xiàn)短暫的卡頓現(xiàn)象氛堕。一般需要在時間上加一個隨機(jī)值,使得過期時間分散一些野蝇。
39讼稚、使用過 Redis 做異步隊(duì)列么,你是怎么用的绕沈?
答:一般使用 list 結(jié)構(gòu)作為隊(duì)列乱灵,rpush 生產(chǎn)消息,lpop 消費(fèi)消息七冲。當(dāng) lpop 沒有消息的時候,要適當(dāng) sleep 一會再重試规婆。如果對方追問可不可以不用 sleep 呢澜躺?list 還有個指令叫 blpop蝉稳,在沒有消息的時候掘鄙,它會阻塞住直到消息到來耘戚。如果對方追問能不能生產(chǎn)一次消費(fèi)多次呢操漠?使用 pub/sub 主題訂閱者模式,可以實(shí)現(xiàn)1:N 的消息隊(duì)列浊伙。
如果對方追問 pub/sub 有什么缺點(diǎn)撞秋?
在消費(fèi)者下線的情況下嚣鄙,生產(chǎn)的消息會丟失,得使用專業(yè)的消息隊(duì)列如 RabbitMQ等哑子。
如果對方追問 redis 如何實(shí)現(xiàn)延時隊(duì)列舅列?
我估計(jì)現(xiàn)在你很想把面試官一棒打死如果你手上有一根棒球棍的話,怎么問的這么詳細(xì)卧蜓。但是你很克制,然后神態(tài)自若的回答道:使用 sortedset榨惠,拿時間戳作為score其爵,消息內(nèi)容作為 key 調(diào)用 zadd 來生產(chǎn)消息,消費(fèi)者用 zrangebyscore 指令獲取 N 秒之前的數(shù)據(jù)輪詢進(jìn)行處理简烤。到這里,面試官暗地里已經(jīng)對你豎起了大拇指横侦。但是他不知道的是此刻你卻豎起了中指绰姻,在椅子背后。
40榨馁、使用過 Redis 分布式鎖么帜矾,它是什么回事屑柔?
先拿 setnx 來爭搶鎖掸宛,搶到之后招拙,再用 expire 給鎖加一個過期時間防止鎖忘記了釋放。
這時候?qū)Ψ綍嬖V你說你回答得不錯饰序,然后接著問如果在 setnx 之后執(zhí)行 expire之前進(jìn)程意外 crash 或者要重啟維護(hù)了闻妓,那會怎么樣?這時候你要給予驚訝的反饋:唉注祖,是喔均唉,這個鎖就永遠(yuǎn)得不到釋放了。緊接著你需要抓一抓自己得腦袋罩缴,故作思考片刻层扶,好像接下來的結(jié)果是你主動思考出來的,然后回答:我記得 set 指令有非常復(fù)雜的參數(shù)檬寂,這個應(yīng)該是可以同時把 setnx 和expire 合成一條指令來用的戳表!對方這時會顯露笑容,心里開始默念:摁镣屹,這小子還不錯价涝。