本篇內(nèi)容介紹了“redis是單線程的嗎”的有關(guān)知識,在實(shí)際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細(xì)閱讀,能夠?qū)W有所成!
10多年的明溪網(wǎng)站建設(shè)經(jīng)驗(yàn),針對設(shè)計、前端、開發(fā)、售后、文案、推廣等六對一服務(wù),響應(yīng)快,48小時及時工作處理。營銷型網(wǎng)站的優(yōu)勢是能夠根據(jù)用戶設(shè)備顯示端的尺寸不同,自動調(diào)整明溪建站的顯示方式,使網(wǎng)站能夠適用不同顯示終端,在瀏覽器中調(diào)整網(wǎng)站的寬度,無論在任何一種瀏覽器上瀏覽網(wǎng)站,都能展現(xiàn)優(yōu)雅布局與設(shè)計,從而大程度地提升瀏覽體驗(yàn)。創(chuàng)新互聯(lián)建站從事“明溪網(wǎng)站設(shè)計”,“明溪網(wǎng)站推廣”以來,每個客戶項(xiàng)目都認(rèn)真落實(shí)執(zhí)行。
一、前言
近乎所有與Java相關(guān)的面試都會問到緩存的問題,基礎(chǔ)一點(diǎn)的會問到什么是“二八定律”、什么是“熱數(shù)據(jù)和冷數(shù)據(jù)” ,復(fù)雜一點(diǎn)的會問到緩存雪崩、緩存穿透、緩存預(yù)熱、緩存更新、緩存降級等問題,這些看似不常見的概念,都與我們的緩存服務(wù)器相關(guān),一般常用的緩存服務(wù)器有Redis、Memcached等,而筆者目前最常用的也只有Redis這一種。
如果你在以前面試的時候還沒有遇到過面試官問你:為什么說Redis是單線程或者Redis為什么這么快?,那么你看到這篇文章的時候,你應(yīng)該覺得是一件很幸運(yùn)的事情!如果你剛好是一位高逼格的面試官,你也可以拿這道題去面試對面“望穿秋水”般的小伙伴,測試一下他的掌握程度。
好啦!步入正題!我們先探討一下Redis是什么,Redis為什么這么快、然后在探討一下為什么Redis是單線程的?
Redis是一個開源的內(nèi)存中的數(shù)據(jù)結(jié)構(gòu)存儲系統(tǒng),它可以用作:數(shù)據(jù)庫、緩存和消息中間件。
它支持多種類型的數(shù)據(jù)結(jié)構(gòu),如字符串(Strings),散列(Hash),列表(List),集合(Set),有序集合(Sorted Set或者是ZSet)與范圍查詢,Bitmaps,Hyperloglogs 和地理空間(Geospatial)索引半徑查詢。其中常見的數(shù)據(jù)結(jié)構(gòu)類型有:String、List、Set、Hash、ZSet這5種。
Redis 內(nèi)置了復(fù)制(Replication),LUA腳本(Lua scripting), LRU驅(qū)動事件(LRU eviction),事務(wù)(Transactions) 和不同級別的磁盤持久化(Persistence),并通過 Redis哨兵(Sentinel)和自動分區(qū)(Cluster)提供高可用性(High Availability)。
Redis也提供了持久化的選項(xiàng),這些選項(xiàng)可以讓用戶將自己的數(shù)據(jù)保存到磁盤上面進(jìn)行存儲。根據(jù)實(shí)際情況,可以每隔一定時間將數(shù)據(jù)集導(dǎo)出到磁盤(快照),或者追加到命令日志中(AOF只追加文件),他會在執(zhí)行寫命令時,將被執(zhí)行的寫命令復(fù)制到硬盤里面。您也可以關(guān)閉持久化功能,將Redis作為一個高效的網(wǎng)絡(luò)的緩存數(shù)據(jù)功能使用。
Redis不使用表,他的數(shù)據(jù)庫不會預(yù)定義或者強(qiáng)制去要求用戶對Redis存儲的不同數(shù)據(jù)進(jìn)行關(guān)聯(lián)。
數(shù)據(jù)庫的工作模式按存儲方式可分為:硬盤數(shù)據(jù)庫和內(nèi)存數(shù)據(jù)庫。Redis 將數(shù)據(jù)儲存在內(nèi)存里面,讀寫數(shù)據(jù)的時候都不會受到硬盤 I/O 速度的限制,所以速度極快。
1、硬盤數(shù)據(jù)庫的工作模式:
2、內(nèi)存數(shù)據(jù)庫的工作模式:
看完上述的描述,對于一些常見的Redis相關(guān)的面試題,是否有所認(rèn)識了,例如:什么是Redis、Redis常見的數(shù)據(jù)結(jié)構(gòu)類型有哪些、Redis是如何進(jìn)行持久化的等。
Redis采用的是基于內(nèi)存的采用的是單進(jìn)程單線程模型的 KV 數(shù)據(jù)庫,由C語言編寫,官方提供的數(shù)據(jù)是可以達(dá)到100000+的QPS(每秒內(nèi)查詢次數(shù))。這個數(shù)據(jù)不比采用單進(jìn)程多線程的同樣基于內(nèi)存的 KV 數(shù)據(jù)庫 Memcached 差!有興趣的可以參考官方的基準(zhǔn)程序測試:https://redis.io/topics/benchmarks
橫軸是連接數(shù),縱軸是QPS。此時,這張圖反映了一個數(shù)量級,希望大家在面試的時候可以正確的描述出來,不要問你的時候,你回答的數(shù)量級相差甚遠(yuǎn)!
1、完全基于內(nèi)存,絕大部分請求是純粹的內(nèi)存操作,非??焖?。數(shù)據(jù)存在內(nèi)存中,類似于HashMap,HashMap的優(yōu)勢就是查找和操作的時間復(fù)雜度都是O(1);
2、數(shù)據(jù)結(jié)構(gòu)簡單,對數(shù)據(jù)操作也簡單,Redis中的數(shù)據(jù)結(jié)構(gòu)是專門進(jìn)行設(shè)計的;
3、采用單線程,避免了不必要的上下文切換和競爭條件,也不存在多進(jìn)程或者多線程導(dǎo)致的切換而消耗 CPU,不用去考慮各種鎖的問題,不存在加鎖釋放鎖操作,沒有因?yàn)榭赡艹霈F(xiàn)死鎖而導(dǎo)致的性能消耗;
4、使用多路I/O復(fù)用模型,非阻塞IO;
5、使用底層模型不同,它們之間底層實(shí)現(xiàn)方式以及與客戶端之間通信的應(yīng)用協(xié)議不一樣,Redis直接自己構(gòu)建了VM 機(jī)制 ,因?yàn)橐话愕南到y(tǒng)調(diào)用系統(tǒng)函數(shù)的話,會浪費(fèi)一定的時間去移動和請求;
以上幾點(diǎn)都比較好理解,下邊我們針對多路 I/O 復(fù)用模型進(jìn)行簡單的探討:
(1)多路 I/O 復(fù)用模型
多路I/O復(fù)用模型是利用 select、poll、epoll 可以同時監(jiān)察多個流的 I/O 事件的能力,在空閑的時候,會把當(dāng)前線程阻塞掉,當(dāng)有一個或多個流有 I/O 事件時,就從阻塞態(tài)中喚醒,于是程序就會輪詢一遍所有的流(epoll 是只輪詢那些真正發(fā)出了事件的流),并且只依次順序的處理就緒的流,這種做法就避免了大量的無用操作。
這里“多路”指的是多個網(wǎng)絡(luò)連接,“復(fù)用”指的是復(fù)用同一個線程。采用多路 I/O 復(fù)用技術(shù)可以讓單個線程高效的處理多個連接請求(盡量減少網(wǎng)絡(luò) IO 的時間消耗),且 Redis 在內(nèi)存中操作數(shù)據(jù)的速度非??欤簿褪钦f內(nèi)存內(nèi)的操作不會成為影響Redis性能的瓶頸,主要由以上幾點(diǎn)造就了 Redis 具有很高的吞吐量。
我們首先要明白,上邊的種種分析,都是為了營造一個Redis很快的氛圍!官方FAQ表示,因?yàn)镽edis是基于內(nèi)存的操作,CPU不是Redis的瓶頸,Redis的瓶頸最有可能是機(jī)器內(nèi)存的大小或者網(wǎng)絡(luò)帶寬。既然單線程容易實(shí)現(xiàn),而且CPU不會成為瓶頸,那就順理成章地采用單線程的方案了(畢竟采用多線程會有很多麻煩!)。
可以參考:https://redis.io/topics/faq
看到這里,你可能會氣哭!本以為會有什么重大的技術(shù)要點(diǎn)才使得Redis使用單線程就可以這么快,沒想到就是一句官方看似糊弄我們的回答!但是,我們已經(jīng)可以很清楚的解釋了為什么Redis這么快,并且正是由于在單線程模式的情況下已經(jīng)很快了,就沒有必要在使用多線程了!
但是,我們使用單線程的方式是無法發(fā)揮多核CPU 性能,不過我們可以通過在單機(jī)開多個Redis 實(shí)例來完善!
警告1:這里我們一直在強(qiáng)調(diào)的單線程,只是在處理我們的網(wǎng)絡(luò)請求的時候只有一個線程來處理,一個正式的Redis Server運(yùn)行的時候肯定是不止一個線程的,這里需要大家明確的注意一下!例如Redis進(jìn)行持久化的時候會以子進(jìn)程或者子線程的方式執(zhí)行(具體是子線程還是子進(jìn)程待讀者深入研究);例如我在測試服武器上查看Redis進(jìn)程,然后找到該進(jìn)程下的線程:
ps命令的“-T”參數(shù)表示顯示線程(Show threads, possibly with SPID column.)“SID”欄表示線程ID,而“CMD”欄則顯示了線程名稱。
警告2:在上圖中FAQ中的最后一段,表述了從Redis 4.0版本開始會支持多線程的方式,但是,只是在某一些操作上進(jìn)行多線程的操作!所以該篇文章在以后的版本中是否還是單線程的方式需要讀者考證!
以下也是你應(yīng)該知道的幾種模型,祝你的面試一臂之力!
1、單進(jìn)程多線程模型:MySQL、Memcached、Oracle(Windows版本);
2、多進(jìn)程模型:Oracle(Linux版本);
3、Nginx有兩類進(jìn)程,一類稱為Master進(jìn)程(相當(dāng)于管理進(jìn)程),另一類稱為Worker進(jìn)程(實(shí)際工作進(jìn)程)。啟動方式有兩種:
(1)單進(jìn)程啟動:此時系統(tǒng)中僅有一個進(jìn)程,該進(jìn)程既充當(dāng)Master進(jìn)程的角色,也充當(dāng)Worker進(jìn)程的角色。
(2)多進(jìn)程啟動:此時系統(tǒng)有且僅有一個Master進(jìn)程,至少有一個Worker進(jìn)程工作。
(3)Master進(jìn)程主要進(jìn)行一些全局性的初始化工作和管理Worker的工作;事件處理是在Worker中進(jìn)行的。
“Redis是單線程的嗎”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識可以關(guān)注創(chuàng)新互聯(lián)網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實(shí)用文章!