色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

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

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

3天內不再提示

探究Redis網絡模型究竟有多強大(中)

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

3.2.1 創建socket

創建socket這一步和客戶端沒啥區別,不同的是這個socket我們稱之為 等待連接socket(或監聽socket)

3.2.2 綁定端口

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

3.2.3 listen()的真正作用

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

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

我們剛說過監聽socket和客戶端創建的socket沒什么區別,問題就出在這個沒什么區別上。

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

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

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

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

  • 已完成連接隊列

每個已完成TCP3次握手的客戶端連接對應的socket就放在這里,這些socket的狀態為ESTABLISHED

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

圖片

listen與3次握手

解釋一下動圖中的內容:

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

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

3.3 答疑時間

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

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

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

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

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

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

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

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

3.4 BIO究竟阻塞在哪里

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

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

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

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

3.5 改造BIO

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

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

        try {
            serverSocket = new ServerSocket(8099);
            System.out.println("啟動服務:監聽端口:8099");
            // 等待客戶端的連接過來,如果沒有連接過來,就會阻塞
            while (true) {
                // 表示阻塞等待監聽一個客戶端連接,返回的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("收到客戶端發送的消息:" + clientStr);

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

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

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

圖片

BIO改造之后

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

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

Java沒這個能力!

3.6 掀開BIO的遮羞布

接下來帶大家看一下改造之后的BIO代碼在底層都調用了哪一些系統調用,讓我們在底層上對上文的內容加深一下理解。

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

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

straceLinux上的一個程序,該程序可以追蹤并記錄參數后邊運行的進程對內核進行了哪些系統調用。

strace -ff -o out java BIOServerSocketWithThread

其中:

  • -o:

將系統調用的追蹤信息輸出到out文件中,不加這個參數,默認會輸出到標準錯誤stderr

  • -ff

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

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

圖片

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

圖片

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

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

額外說兩點:

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

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

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

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

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

nc localhost 8099

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

圖片

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

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

圖片

發現子進程阻塞在了recvfrom()這個系統調用上,對應的Java源碼就是bufferedReader.readLine();,直到客戶端主動給服務端發送消息,阻塞才會結束。

3.7 BIO總結

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

圖片

BIO模型

BIO之所以是BIO,是因為系統底層調用是阻塞的,上圖中的進程調用recv,其系統調用直到數據包準備好并且被復制到應用程序的緩沖區或者發生錯誤為止才會返回,在此整個期間,進程是被阻塞的,啥也干不了。

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

    關注

    0

    文章

    212

    瀏覽量

    34794
  • 函數
    +關注

    關注

    3

    文章

    4344

    瀏覽量

    62847
  • Redis
    +關注

    關注

    0

    文章

    377

    瀏覽量

    10905
收藏 人收藏

    評論

    相關推薦

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

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

    Redis Stream應用案例

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

    液晶PC與液晶電視究竟有什么區別?

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

    請問一下RFID與NFC究竟有什么關系?

    RFID與NFC究竟有什么關系?
    發表于 06-15 07:06

    面向列的HBase存儲結構究竟有什么樣的不同之處呢?

    HBase是什么?HBase的存儲結構究竟是怎樣的呢?面向列的HBase存儲結構究竟有什么樣的不同之處呢?
    發表于 06-16 06:52

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

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

    PCI-E4.0究竟有什么優勢?

    PCI-E4.0究竟有什么優勢?PCI-E究竟指的是什么呢?
    發表于 06-18 06:54

    內存時序究竟有多重要呢?究竟該如何去選擇內存條呢?

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

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

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

    OpenPLC開源工業控制器究竟有何用處

    OpenPLC開源工業控制器有哪些優點?OpenPLC開源工業控制器有哪些功能?OpenPLC開源工業控制器究竟有何用處?
    發表于 09-02 07:42

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

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

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

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

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

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

    探究Redis網絡模型究竟有多強大(上)

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

    探究Redis網絡模型究竟有多強大(下)

    接下來的非阻塞IO我們只抓主要矛盾,其余參考BIO即可。 如果你看過其他介紹非阻塞IO的文
    的頭像 發表于 03-03 09:50 ?428次閱讀
    <b class='flag-5'>探究</b><b class='flag-5'>Redis</b><b class='flag-5'>網絡</b><b class='flag-5'>模型</b><b class='flag-5'>究竟有多強大</b>(下)
    主站蜘蛛池模板: 甜性涩爱快播| 亚洲手机在线人成视频| 夜夜澡人人爽人人喊_欧美| 国产精品高清视频在线| 乳色吐息未增删樱花ED在线观看| a级成人免费毛片完整版| 男女午夜性爽快免费视频不卡| 稚嫩挤奶h调教h| 久久内在线视频精品mp4| 伊人久久综在合线影院| 久久久96| 2022久久精品国产色蜜蜜麻豆| 蜜桃传媒一区二区亚洲AV| 97久久伊人精品影院| 女子叉开腿让男子桶免费软件| 9位美女厕所撒尿11分| 欧美一区二区视频97色伦| 成人国内精品久久久久影| 色翁荡熄月月| 国产欧美国产综合第一区| 亚洲人成网77777色在线播放| 娇喘高潮教室h| 777EY_卡通动漫_1页| 欧美精品一区二区三区四区| A级超碰视频在线观看| 日日操夜夜操狠狠操| 国产精品VIDEOSSEX久久发布| 亚洲AV无码国产精品午夜久久| 黄色片网站下载| 66美女人体| 色色噜一噜| 精品亚洲一区二区三区在线播放 | 午夜福利免费体检区| 國産麻豆AVMDMD0179| 中文字幕不卡一区二区三区| 欧美日韩黄色| 国产精品免费久久久久影院| 一个人HD在线观看免费高清视频 | 最新果冻传媒在线观看免费版| 奇虎成人网| 国产人妻人伦精品久久久|