欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Netty源碼分析NioEventLoop處理IO事件相關(guān)邏輯

 更新時(shí)間:2022年03月25日 15:03:25   作者:向南是個(gè)萬(wàn)人迷  
這篇文章主要介紹了Netty源碼分析NioEventLoop處理IO事件相關(guān)邏輯,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

前文傳送門(mén):NioEventLoop執(zhí)行select操作入口

之前我們了解了執(zhí)行select()操作的相關(guān)邏輯, 這一小節(jié)我們繼續(xù)學(xué)習(xí)輪詢(xún)到io事件的相關(guān)邏輯:

NioEventLoop的run()方法:

protected void run() {
    for (;;) {
        try {
            switch (selectStrategy.calculateStrategy(selectNowSupplier, hasTasks())) {
                case SelectStrategy.CONTINUE:
                    continue;
                case SelectStrategy.SELECT:
                    //輪詢(xún)io事件(1)
                    select(wakenUp.getAndSet(false));
                    if (wakenUp.get()) {
                        selector.wakeup();
                    }
                default:
            }
            cancelledKeys = 0;
            needsToSelectAgain = false;
            //默認(rèn)是50
            final int ioRatio = this.ioRatio; 
            if (ioRatio == 100) {
                try {
                    processSelectedKeys();
                } finally {
                    runAllTasks();
                }
            } else {
                //記錄下開(kāi)始時(shí)間
                final long ioStartTime = System.nanoTime();
                try {
                    //處理輪詢(xún)到的key(2)
                    processSelectedKeys();
                } finally {
                    //計(jì)算耗時(shí)
                    final long ioTime = System.nanoTime() - ioStartTime;
                    //執(zhí)行task(3)
                    runAllTasks(ioTime * (100 - ioRatio) / ioRatio);
                }
            }
        } catch (Throwable t) {
            handleLoopException(t);
        }
        //代碼省略
    }
}

我們首先看 if (ioRatio == 100) 這個(gè)判斷, ioRatio主要是用來(lái)控制processSelectedKeys()方法執(zhí)行時(shí)間和任務(wù)隊(duì)列執(zhí)行時(shí)間的比例, 其中ioRatio默認(rèn)是50, 所以會(huì)走到下一步else

首先通過(guò) final long ioStartTime = System.nanoTime() 記錄下開(kāi)始時(shí)間, 再通過(guò)processSelectedKeys()方法處理輪詢(xún)到的key

processSelectedKeys()方法

private void processSelectedKeys() { 
    if (selectedKeys != null) {
        //flip()方法會(huì)直接返回key的數(shù)組
        processSelectedKeysOptimized(selectedKeys.flip());
    } else {
        processSelectedKeysPlain(selector.selectedKeys());
    }
}

我們知道selector通過(guò)netty優(yōu)化之后, 會(huì)初始化 selectedKeys這個(gè)屬性, 所以這個(gè)屬性不為空就會(huì)走到 processSelectedKeysOptimized(selectedKeys.flip()) 方法, 這個(gè)方法就是對(duì)應(yīng)優(yōu)化過(guò)的selector進(jìn)行操作的

如果是非優(yōu)化的selector, 則會(huì)進(jìn)入 processSelectedKeysPlain(selector.selectedKeys()) 方法

selectedKeys.flip()為selectedKey中綁定的數(shù)組, 我們之前小節(jié)講過(guò)selectedKeys其實(shí)是通過(guò)數(shù)組存儲(chǔ)的, 所以經(jīng)過(guò)select()操作如果監(jiān)聽(tīng)到事件selectedKeys的數(shù)組就會(huì)有值

processSelectedKeysOptimized(selectedKeys.flip())方法

private void processSelectedKeysOptimized(SelectionKey[] selectedKeys) {
    //通過(guò)for循環(huán)遍歷數(shù)組
    for (int i = 0;; i ++) {
        //拿到當(dāng)前的selectionKey
        final SelectionKey k = selectedKeys[i];
        if (k == null) {
            break;
        }
        //將當(dāng)前引用設(shè)置為null
        selectedKeys[i] = null;
        //獲取channel(NioSeverSocketChannel)
        final Object a = k.attachment();
        //如果是AbstractNioChannel, 則調(diào)用processSelectedKey()方法處理io事件
        if (a instanceof AbstractNioChannel) {
            processSelectedKey(k, (AbstractNioChannel) a);
        } else {
            @SuppressWarnings("unchecked")
            NioTask<SelectableChannel> task = (NioTask<SelectableChannel>) a;
            processSelectedKey(k, task);
        }

        //代碼省略
    }
}

首先通過(guò)for循環(huán)遍歷數(shù)組中的每一個(gè)key, 獲得key之后首先將數(shù)組中對(duì)應(yīng)的下標(biāo)清空, 因?yàn)閟elector不會(huì)自動(dòng)清空, 這與我們使用原生selector時(shí)候, 通過(guò)遍歷selector.selectedKeys()的set的時(shí)候, 拿到key之后要執(zhí)行remove()是一個(gè)意思

之后獲取注冊(cè)在key上的channel, 判斷channel是不是AbstractNioChannel, 通常情況都是AbstractNioChannel, 所以這里會(huì)執(zhí)行 processSelectedKey(k, (AbstractNioChannel) a) 

processSelectedKey(k, (AbstractNioChannel) a)方法

private void processSelectedKey(SelectionKey k, AbstractNioChannel ch) {
    //獲取到channel中的unsafe
    final AbstractNioChannel.NioUnsafe unsafe = ch.unsafe();
    //如果這個(gè)key不是合法的, 說(shuō)明這個(gè)channel可能有問(wèn)題
    if (!k.isValid()) {
        //代碼省略
    }
    try {
        //如果是合法的, 拿到key的io事件
        int readyOps = k.readyOps();
        //鏈接事件
        if ((readyOps & SelectionKey.OP_CONNECT) != 0) {
            int ops = k.interestOps();
            ops &= ~SelectionKey.OP_CONNECT;
            k.interestOps(ops);
            unsafe.finishConnect();
        }
        //寫(xiě)事件
        if ((readyOps & SelectionKey.OP_WRITE) != 0) {
            ch.unsafe().forceFlush();
        }
        //讀事件和接受鏈接事件
        //如果當(dāng)前NioEventLoop是work線程的話(huà), 這里就是op_read事件
        //如果是當(dāng)前NioEventLoop是boss線程的話(huà), 這里就是op_accept事件
        if ((readyOps & (SelectionKey.OP_READ | SelectionKey.OP_ACCEPT)) != 0 || readyOps == 0) {
            unsafe.read();
            if (!ch.isOpen()) {
                return;
            }
        }
    } catch (CancelledKeyException ignored) {
        unsafe.close(unsafe.voidPromise());
    }
}

我們首先獲取和channel綁定的unsafe, 之后拿到channel注冊(cè)的事件

我們關(guān)注

 if ((readyOps & (SelectionKey.OP_READ | SelectionKey.OP_ACCEPT)) != 0 || readyOps == 0) 

這個(gè)判斷, 這個(gè)判斷相信注釋上寫(xiě)的很明白, 如果當(dāng)前NioEventLoop是work線程的話(huà), 這里就是op_read事件, 如果是當(dāng)前NioEventLoop是boss線程的話(huà), 這里就是op_accept事件

然后會(huì)通過(guò)channel綁定的unsafe對(duì)象執(zhí)行read()方法用于處理鏈接或者讀寫(xiě)事件

以上就是NioEventLoop對(duì)io事件的處理過(guò)程, 有關(guān)read()方法執(zhí)行邏輯, 會(huì)在以后的章節(jié)中詳細(xì)剖析,更多關(guān)于Netty NioEventLoop處理IO事件邏輯的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Java編程實(shí)現(xiàn)提取文章中關(guān)鍵字的方法

    Java編程實(shí)現(xiàn)提取文章中關(guān)鍵字的方法

    這篇文章主要介紹了Java編程實(shí)現(xiàn)提取文章中關(guān)鍵字的方法,較為詳細(xì)的分析了Java提取文章關(guān)鍵字的原理與具體實(shí)現(xiàn)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下
    2015-11-11
  • ES修改字段類(lèi)型的操作方式

    ES修改字段類(lèi)型的操作方式

    ES修改字段類(lèi)型是指在已有的索引中,通過(guò)特定的操作方式將某個(gè)字段的類(lèi)型修改為其它類(lèi)型,這篇文章主要介紹了ES修改字段類(lèi)型的相關(guān)知識(shí),感興趣的朋友跟隨小編一起看看吧
    2023-10-10
  • SpringBoot使用flyway初始化數(shù)據(jù)庫(kù)

    SpringBoot使用flyway初始化數(shù)據(jù)庫(kù)

    這篇文章主要介紹了SpringBoot如何使用flyway初始化數(shù)據(jù)庫(kù),幫助大家更好的理解和學(xué)習(xí)使用SpringBoot框架,感興趣的朋友可以了解下
    2021-03-03
  • springboot與vue實(shí)現(xiàn)簡(jiǎn)單的CURD過(guò)程詳析

    springboot與vue實(shí)現(xiàn)簡(jiǎn)單的CURD過(guò)程詳析

    這篇文章主要介紹了springboot與vue實(shí)現(xiàn)簡(jiǎn)單的CURD過(guò)程詳析,圍繞springboot與vue的相關(guān)資料展開(kāi)實(shí)現(xiàn)CURD過(guò)程的過(guò)程介紹,需要的小伙伴可以參考一下
    2022-01-01
  • Java String類(lèi)的常用方法匯總

    Java String類(lèi)的常用方法匯總

    這篇文章主要為大家詳細(xì)匯總了Java String類(lèi)的常用方法,感興趣的小伙伴們可以參考一下
    2016-07-07
  • java使用dom4j解析xml配置文件實(shí)現(xiàn)抽象工廠反射示例

    java使用dom4j解析xml配置文件實(shí)現(xiàn)抽象工廠反射示例

    本文主要介紹了java使用dom4j讀取配置文件實(shí)現(xiàn)抽象工廠和反射的示例,在Java中也可以同Donet一樣,將差異配置在配置文件里面。另外,我們采用下面的方式實(shí)現(xiàn),將會(huì)更加便捷
    2014-01-01
  • java基本教程之多線程基本概念 java多線程教程

    java基本教程之多線程基本概念 java多線程教程

    多線程是Java中不可避免的一個(gè)重要主體。下面是對(duì)“JDK中新增JUC包”之前的Java多線程內(nèi)容的講解,JUC包是由Java大師Doug Lea完成并在JDK1.5版本添加到Java中的
    2014-01-01
  • java 自定義可繼承枚舉Enum的案例

    java 自定義可繼承枚舉Enum的案例

    這篇文章主要介紹了java 自定義可繼承枚舉Enum的案例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2021-02-02
  • Spring Cloud Feign的文件上傳實(shí)現(xiàn)的示例代碼

    Spring Cloud Feign的文件上傳實(shí)現(xiàn)的示例代碼

    這篇文章主要介紹了Spring Cloud Feign的文件上傳實(shí)現(xiàn)的示例代碼,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2018-03-03
  • Java的MyBatis框架項(xiàng)目搭建與hellow world示例

    Java的MyBatis框架項(xiàng)目搭建與hellow world示例

    MyBatis框架為Java程序的數(shù)據(jù)庫(kù)操作帶來(lái)了很大的便利,這里我們就從最基礎(chǔ)的入手,來(lái)看一下Java的MyBatis框架項(xiàng)目搭建與hellow world示例,需要的朋友可以參考下
    2016-06-06

最新評(píng)論