0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

探究Redis網(wǎng)絡(luò)模型究竟有多強大(中)

jf_78858299 ? 來源:蟬沐風的碼場 ? 作者:蟬沐風 ? 2023-03-03 09:49 ? 次閱讀

3.2.1 創(chuàng)建socket

創(chuàng)建socket這一步和客戶端沒啥區(qū)別,不同的是這個socket我們稱之為 等待連接socket(或監(jiān)聽socket) 。

3.2.2 綁定端口

bind()函數(shù)會將端口號寫入上一步生成的監(jiān)聽socket中,這樣一來,監(jiān)聽socket就完整保存了服務(wù)端的IP端口號。

3.2.3 listen()的真正作用

listen(<Server描述符>, <最大連接數(shù)>);

很多小伙伴一定會對這個listen()有疑問,監(jiān)聽socket都已經(jīng)創(chuàng)建完了,端口也已經(jīng)綁定完了,為什么還要多調(diào)用一個listen()呢?

我們剛說過監(jiān)聽socket和客戶端創(chuàng)建的socket沒什么區(qū)別,問題就出在這個沒什么區(qū)別上。

socket被創(chuàng)建出來的時候都默認是一個 主動socket ,也就說,內(nèi)核會認為這個socket之后某個時候會調(diào)用connect()主動向別的設(shè)備發(fā)起連接。這個默認對客戶端socket來說很合理,但是監(jiān)聽socket可不行,它只能等著客戶端連接自己,因此我們需要調(diào)用listen()將監(jiān)聽socket從主動設(shè)置為被動,明確告訴內(nèi)核:你要接受指向這個監(jiān)聽socket的連接請求!

此外,listen()的第2個參數(shù)也大有來頭!監(jiān)聽socket真正接受的應(yīng)該是已經(jīng)完整完成3次握手的客戶端,那么還沒完成的怎么辦?總得找個地方放著吧。于是內(nèi)核為每一個監(jiān)聽socket都維護了兩個隊列:

  • 半連接隊列(未完成連接的隊列)

這里存放著暫未徹底完成3次握手的socket(為了防止半連接攻擊,這里存放的其實是占用內(nèi)存極小的request _sock,但是我們直接理解成socket就行了),這些socket的狀態(tài)稱為SYN_RCVD。

  • 已完成連接隊列

每個已完成TCP3次握手的客戶端連接對應(yīng)的socket就放在這里,這些socket的狀態(tài)為ESTABLISHED。

文字太多了,有點干,上個圖!

圖片

listen與3次握手

解釋一下動圖中的內(nèi)容:

  1. 客戶端調(diào)用connect()函數(shù),開始3次握手,首先發(fā)送一個SYN X的報文(X是個數(shù)字,下同);
  2. 服務(wù)端收到來自客戶端的SYN,然后在監(jiān)聽socket對應(yīng)的半連接隊列中創(chuàng)建一個新的socket,然后對客戶端發(fā)回響應(yīng)SYN Y,捎帶手對客戶端的報文給個ACK;
  3. 直到客戶端完成第3次握手,剛才新創(chuàng)建的socket就會被轉(zhuǎn)移到已連接隊列;
  4. 當進程調(diào)用accept()時,會將已連接隊列頭部的socket返回;如果已連接隊列為空,那么進程將被睡眠,直到已連接隊列中有新的socket,進程才會被喚醒,將這個socket返回

第4步就是阻塞的本質(zhì)啊,朋友們!

3.3 答疑時間

Q1.隊列中的對象是socket嗎?

呃。。。乖,咱就把它當成socket就好了,這樣容易理解,其實具體里邊存放的數(shù)據(jù)結(jié)構(gòu)是啥,我也很想知道,等我寫完這篇文章,我研究完了告訴你。

Q2.accept()這個函數(shù)你還沒講是啥意思呢?

accept()函數(shù)是由服務(wù)端調(diào)用的,用于從已連接隊列中返回一個socket描述符;如果socket為阻塞式的,那么如果已連接隊列為空,accept()進程就會被睡眠。BIO恰好就是這個樣子。

Q3.accept()為什么不直接把監(jiān)聽socket返回呢?

因為在隊列中的socket經(jīng)過3次握手過程的控制信息交換,socket的4元組的信息已經(jīng)完整了,用做socket完全沒問題。

監(jiān)聽socket就像一個客服,我們給客服打電話,然后客服找到解決問題的人,幫助我們和解決問題的人建立聯(lián)系,如果直接把監(jiān)聽socket返回,而不使用連接socket,就沒有socket繼續(xù)等待連接了。

哦對了,accept()返回的socket也有個名字,叫 連接socket 。

3.4 BIO究竟阻塞在哪里

拿Server端的BIO來說明這個問題,阻塞在了serverSocket.accept()以及bufferedReader.readLine()這兩個地方。有什么辦法可以證明阻塞嗎?

簡單的很!你在serverSocket.accept(); 的下一行打個斷點,然后debug模式運行BIOServerSocket,在沒有客戶端連接的情況下,這個斷點絕不會觸發(fā)!同樣,在bufferedReader.readLine();下一行打個斷點,在已連接的客戶端發(fā)送數(shù)據(jù)之前,這個斷點絕不會觸發(fā)!

readLine()的阻塞還帶來一個非常嚴重的問題,如果已經(jīng)連接的客戶端一直不發(fā)送消息,readLine()進程就會一直阻塞(處于睡眠狀態(tài)),結(jié)果就是代碼不會再次運行到accept(),這個ServerSocket沒辦法接受新的客戶端連接。

解決這個問題的核心就是別讓代碼卡在readLine()就可以了,我們可以使用新的線程來readLine(),這樣代碼就不會阻塞在readLine()上了。

3.5 改造BIO

改造之后的BIO長這樣,這下子服務(wù)端就可以隨時接受客戶端的連接了,至于啥時候能read到客戶端的數(shù)據(jù),那就讓線程去處理這個事情吧。

public class BIOServerSocketWithThread {
    public static void main(String[] args) {
        ServerSocket serverSocket = null;

        try {
            serverSocket = new ServerSocket(8099);
            System.out.println("啟動服務(wù):監(jiān)聽端口:8099");
            // 等待客戶端的連接過來,如果沒有連接過來,就會阻塞
            while (true) {
                // 表示阻塞等待監(jiān)聽一個客戶端連接,返回的socket表示連接的客戶端信息
                Socket socket = serverSocket.accept(); //連接阻塞
                System.out.println("客戶端:" + socket.getPort());
                // 表示獲取客戶端的請求報文
                new Thread(new Runnable() {
                    @Override
                    public void run() {
                        try {
                            BufferedReader bufferedReader = new BufferedReader(
                                    new InputStreamReader(socket.getInputStream())
                            );
                            String clientStr = bufferedReader.readLine();
                            System.out.println("收到客戶端發(fā)送的消息:" + clientStr);

                            BufferedWriter bufferedWriter = new BufferedWriter(
                                    new OutputStreamWriter(socket.getOutputStream())
                            );
                            bufferedWriter.write("ok\\n");
                            bufferedWriter.flush();
                        } catch (Exception e) {
                            //...
                        }

                    }
                }).start();
            }
        } catch (IOException e) {
            // 錯誤處理
        } finally {
            // 其他處理
        }
    }
}

事情的順利進展不禁讓我們飄飄然,我們居然是使用高階的多線程技術(shù)解決了BIO的阻塞問題,雖然目前每個客戶端都需要一個單獨的線程來處理,但accept()總歸不會被readLine()卡死了。

圖片

BIO改造之后

所以我們改造完之后的程序是不是就是非阻塞IO了呢?

想多了。。。我們只是用了點奇技淫巧罷了,改造完的代碼在系統(tǒng)調(diào)用層面該阻塞的地方還是阻塞,說白了,Java提供的API完全受限于操作系統(tǒng)提供的系統(tǒng)調(diào)用,在Java語言級別沒能力改變底層BIO的事實!

Java沒這個能力!

3.6 掀開BIO的遮羞布

接下來帶大家看一下改造之后的BIO代碼在底層都調(diào)用了哪一些系統(tǒng)調(diào)用,讓我們在底層上對上文的內(nèi)容加深一下理解。

給大家打個氣,接下來的內(nèi)容其實非常好理解,大家跟著文章一步步地走,一定能看得懂,如果自己動手操作一遍,那就更好了。

對了,我下來使用的JDK版本是JDK8。

straceLinux上的一個程序,該程序可以追蹤并記錄參數(shù)后邊運行的進程對內(nèi)核進行了哪些系統(tǒng)調(diào)用。

strace -ff -o out java BIOServerSocketWithThread

其中:

  • -o:

將系統(tǒng)調(diào)用的追蹤信息輸出到out文件中,不加這個參數(shù),默認會輸出到標準錯誤stderr。

  • -ff

如果指定了-o選項,strace會追蹤和程序相關(guān)的每一個進程的系統(tǒng)調(diào)用,并將信息輸出到以進程id為后綴的out文件中。舉個例子,比如BIOServerSocketWithThread程序運行過程中有一個ID為30792的進程,那么該進程的系統(tǒng)調(diào)用日志會輸出到out.30792這個文件中。

我們運行strace命令之后,生成了很多個out文件。

圖片

這么多進程怎么知道哪個是我們需要追蹤的呢?我就挑了一個容量最大的文件進行查看,也就是out.30792,事實上,這個文件也恰好是我們需要的,截取一下里邊的內(nèi)容給大家看一下。

圖片

可以看到圖中的有非常多的行,說明我們寫的這么幾行代碼其實默默調(diào)用了非常多的系統(tǒng)調(diào)用,拋開細枝末節(jié),看一下上圖中我重點標注的系統(tǒng)調(diào)用,是不是就是上文中我解釋過的函數(shù)?我再詳細解釋一下每一步,大家聯(lián)系上文,會對BIO的底層理解的更加通透。

  1. 生成監(jiān)聽socket,并返回socket描述符7,接下來對socket進行操作的函數(shù)都會有一個參數(shù)為7;
  2. 8099端口綁定到監(jiān)聽socket,bind的第一個參數(shù)就是7,說明就是對監(jiān)聽socket進行的操作;
  3. listen()將監(jiān)聽socket(參數(shù)為7)設(shè)置為被動接受連接的socket,并且將隊列的長度設(shè)置為50;
  4. 實際上就是System.out.println("啟動服務(wù):監(jiān)聽端口:8099");這一句的系統(tǒng)調(diào)用,只不過中文被編碼了,所以我特意把:8099圈出來證明一下;

額外說兩點:

其一:可以看到,這么一句簡單的打印輸出在底層實際調(diào)用了兩次write系統(tǒng)調(diào)用,這就是為什么不推薦在生產(chǎn)環(huán)境下使用打印語句的原因,多少會影響系統(tǒng)性能;

其二:write()的第一個參數(shù)為1,也是文件描述符,表示的是標準輸出stdout。

  1. 系統(tǒng)調(diào)用阻塞在了poll()函數(shù),怎么看出來的阻塞?out文件的每一行運行完畢都會有一個 = 返回值,而poll()目前沒有返回值,因此阻塞了。實際上poll()系統(tǒng)調(diào)用對應(yīng)的Java語句就是serverSocket.accept();。

不對???為什么底層調(diào)用的不是accept()而是poll()?poll()應(yīng)該是多路復(fù)用才是啊。在JDK4之前,底層確實直接調(diào)用的是accept(),但是之后的JDK對這一步進行了優(yōu)化,除了調(diào)用accept(),還加上了poll()。poll()的細節(jié)我們下文再說,這里可以起碼證明了poll()函數(shù)依然是阻塞的,所以整個BIO的阻塞邏輯沒有改變。

接下來我們起一個客戶端對程序發(fā)起連接,直接用Linux上的nc程序即可,比較簡單:

nc localhost 8099

發(fā)起連接之后(但并未主動發(fā)送信息),out.30792的內(nèi)容發(fā)生了變化:

圖片

  1. poll()函數(shù)結(jié)束阻塞,程序接著調(diào)用accept()函數(shù)返回一個連接socket,該socket的描述符為8;
  2. 就是System.out.println("客戶端:" + socket.getPort());的底層調(diào)用;
  3. 底層使用clone()創(chuàng)造了一個新進程去處理連接socket,該進程的pid為31168,因此JDK8的線程在底層其實就是輕量級進程;
  4. 回到poll()函數(shù)繼續(xù)阻塞等待新客戶端連接。

由于創(chuàng)建了一個新的進程,因此在目錄下對多出一個out.31168的文件,我們看一下該文件的內(nèi)容:

圖片

發(fā)現(xiàn)子進程阻塞在了recvfrom()這個系統(tǒng)調(diào)用上,對應(yīng)的Java源碼就是bufferedReader.readLine();,直到客戶端主動給服務(wù)端發(fā)送消息,阻塞才會結(jié)束。

3.7 BIO總結(jié)

到此為止,我們就通過底層的系統(tǒng)調(diào)用證明了BIO在accept()以及readLine()上的阻塞。最后用一張圖來結(jié)束BIO之旅。

圖片

BIO模型

BIO之所以是BIO,是因為系統(tǒng)底層調(diào)用是阻塞的,上圖中的進程調(diào)用recv,其系統(tǒng)調(diào)用直到數(shù)據(jù)包準備好并且被復(fù)制到應(yīng)用程序的緩沖區(qū)或者發(fā)生錯誤為止才會返回,在此整個期間,進程是被阻塞的,啥也干不了。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • Socket
    +關(guān)注

    關(guān)注

    0

    文章

    212

    瀏覽量

    34773
  • 函數(shù)
    +關(guān)注

    關(guān)注

    3

    文章

    4341

    瀏覽量

    62800
  • Redis
    +關(guān)注

    關(guān)注

    0

    文章

    376

    瀏覽量

    10898
收藏 人收藏

    評論

    相關(guān)推薦

    小米5的那顆核心,究竟有多強

    小米5的那顆核心,究竟有多強自從小米在2011年崛起后,高通的驍龍系列處理器就逐漸成為了市面上旗艦手機的主流處理器,從蝎子核心到環(huán)蛇核心,再到現(xiàn)在的驍龍810,高通一直在進步。那么,這款驍龍810
    發(fā)表于 06-01 19:35

    Redis Stream應(yīng)用案例

    今天介紹的主角——Redis Stream,本身就是起源于IRC中一個用戶的idea。IRC的模型如下,在某個IRC頻道的用戶,既可以向所有的其他用戶自由的發(fā)送消息,也可以接收其他所有用戶發(fā)送
    發(fā)表于 06-26 17:15

    液晶PC與液晶電視究竟有什么區(qū)別?

    為什么要選擇液晶?液晶PC與液晶電視究竟有什么區(qū)別?如何選擇液晶PC與液晶電視?
    發(fā)表于 06-07 06:13

    請問一下RFID與NFC究竟有什么關(guān)系?

    RFID與NFC究竟有什么關(guān)系?
    發(fā)表于 06-15 07:06

    面向列的HBase存儲結(jié)構(gòu)究竟有什么樣的不同之處呢?

    HBase是什么?HBase的存儲結(jié)構(gòu)究竟是怎樣的呢?面向列的HBase存儲結(jié)構(gòu)究竟有什么樣的不同之處呢?
    發(fā)表于 06-16 06:52

    請問一下芯片制造究竟有多難?

    請問一下芯片制造究竟有多難?
    發(fā)表于 06-18 06:53

    PCI-E4.0究竟有什么優(yōu)勢?

    PCI-E4.0究竟有什么優(yōu)勢?PCI-E究竟指的是什么呢?
    發(fā)表于 06-18 06:54

    內(nèi)存時序究竟有多重要呢?究竟該如何去選擇內(nèi)存條呢?

    內(nèi)存時序究竟有多重要呢?究竟該如何去選擇內(nèi)存條呢?DDR內(nèi)存時序是高一些好還是低一些好?
    發(fā)表于 06-18 08:20

    定時器中斷類型探究 精選資料分享

     一直在用的stm32定時器的中斷都是TIM_IT_Update更新中斷,也沒問為什么,直到碰到有人使用TIM_IT_CC1斷,才想到這定時器的中斷類型究竟有什么區(qū)別,都怪當時學習stm32的時候
    發(fā)表于 08-13 06:28

    OpenPLC開源工業(yè)控制器究竟有何用處

    OpenPLC開源工業(yè)控制器有哪些優(yōu)點?OpenPLC開源工業(yè)控制器有哪些功能?OpenPLC開源工業(yè)控制器究竟有何用處?
    發(fā)表于 09-02 07:42

    華為榮耀Magic今日發(fā)布:“未來”手機究竟有多強

    華為榮耀即將在12月16日發(fā)布最新的“未來”手機magic,關(guān)于這款手機的爆料在今日已經(jīng)鋪天蓋地,今天,小編將為大家整理一下,給大家一個榮耀Magic的基本判斷,看看這款旗艦究竟有多強力!
    發(fā)表于 12-16 09:34 ?3324次閱讀

    ibm的2nm芯片究竟有多強 2nm芯片對續(xù)航的影響

    全球首顆2nm芯片的問世對半導(dǎo)體行業(yè)影響重大,IBM通過與AMD、三星及GlobalFoundries等多家公司的合作,最終抵達了2nm芯片制程的節(jié)點,推出了2nm的測試芯片。那么這顆芯片究竟有多強呢?它對續(xù)航的影響又有多大呢?
    的頭像 發(fā)表于 06-23 09:35 ?2138次閱讀

    Molex莫仕連接器的功能究竟有多強大?看他們的行業(yè)應(yīng)用你就知道了!

    KOYUELEC光與電子:Molex莫仕連接器的功能究竟有多強大?看他們的行業(yè)應(yīng)用你就知道了!
    的頭像 發(fā)表于 12-31 12:30 ?1.1w次閱讀

    探究Redis網(wǎng)絡(luò)模型究竟有多強大(上)

    本文將從BIO開始介紹,經(jīng)過NIO、多路復(fù)用,最終說回Redis的Reactor模型,力求詳盡。本文與其他文章的不同點主要在于:
    的頭像 發(fā)表于 03-03 09:46 ?465次閱讀
    <b class='flag-5'>探究</b><b class='flag-5'>Redis</b><b class='flag-5'>網(wǎng)絡(luò)</b><b class='flag-5'>模型</b><b class='flag-5'>究竟有多強大</b>(上)

    探究Redis網(wǎng)絡(luò)模型究竟有多強大(下)

    接下來的非阻塞IO我們只抓主要矛盾,其余參考BIO即可。 如果你看過其他介紹非阻塞IO的文
    的頭像 發(fā)表于 03-03 09:50 ?422次閱讀
    <b class='flag-5'>探究</b><b class='flag-5'>Redis</b><b class='flag-5'>網(wǎng)絡(luò)</b><b class='flag-5'>模型</b><b class='flag-5'>究竟有多強大</b>(下)