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

Netty組件NioEventLoopGroup創(chuàng)建線程執(zhí)行器源碼解析

 更新時(shí)間:2022年03月25日 15:38:07   作者:向南是個(gè)萬(wàn)人迷  
這篇文章主要介紹了Netty組件NioEventLoopGroup創(chuàng)建線程執(zhí)行器源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

前言

通過(guò)上一章的學(xué)習(xí), 我們了解了Server啟動(dòng)的大致流程, 有很多組件與模塊并沒(méi)有細(xì)講, 從這個(gè)章開始, 我們開始詳細(xì)剖析netty的各個(gè)組件, 并結(jié)合啟動(dòng)流程, 將這些組件的使用場(chǎng)景及流程進(jìn)行一個(gè)詳細(xì)的說(shuō)明

這一章主要學(xué)習(xí)NioEventLoop相關(guān)的知識(shí),何為NioEventLoop? NioEventLoop是netty的一個(gè)線程, 在上一節(jié)我們創(chuàng)建兩個(gè)NioEventLoopGroup:

EventLoopGroup bossGroup = new NioEventLoopGroup(1);
EventLoopGroup workerGroup = new NioEventLoopGroup();

這里創(chuàng)建了兩個(gè)group, 我們提過(guò)這是boss線程組和worker線程組, 其實(shí)這兩個(gè)線程組就相當(dāng)于兩個(gè)NioEventLoop的集合, 默認(rèn)每個(gè)NioEventLoopGroup創(chuàng)建時(shí), 如果不傳入線程數(shù), 會(huì)創(chuàng)建cpu核數(shù)*2個(gè)NioEventLoop線程, 其中boss線程通過(guò)輪詢處理Server的accept事件, 而完成accept事件之后, 就會(huì)創(chuàng)建客戶端channel, 通過(guò)一定的策略, 分發(fā)到worker線程進(jìn)行處理, 而worker線程, 則主要用于處理客戶端的讀寫事件

除了輪詢事件, EventLoop線程還維護(hù)了兩個(gè)隊(duì)列, 一個(gè)是延遲任務(wù)隊(duì)列, 另一個(gè)是普通任務(wù)隊(duì)列, 在進(jìn)行事件輪詢的同時(shí), 如果隊(duì)列中有任務(wù)需要執(zhí)行則會(huì)去執(zhí)行隊(duì)列中的任務(wù)

一個(gè)NioEventLoop綁定一個(gè)selector用于處理多個(gè)客戶端channel, 但是一個(gè)客戶端channel只能被一個(gè)NioEventLoop處理, 具體關(guān)系如圖2-0-1所示:

圖中我們看到, 一個(gè)NioEventLoopGroup下有多個(gè)NioEventLoop線程, 而一個(gè)線程可以處理多個(gè)channel, 其中有個(gè)叫pipeline和handler的東西, 同學(xué)們可能比較陌生, 這是netty的事件傳輸機(jī)制, 每個(gè)pipeline和channel唯一綁定, 這里只需要稍作了解, 之后章節(jié)會(huì)帶大家詳細(xì)剖析

了解了這些概念, 我們繼續(xù)以小節(jié)的形式對(duì)NioEventLoop進(jìn)行剖析

第一節(jié):  NioEventLoopGroup之創(chuàng)建線程執(zhí)行器

首先回到第一章最開始的demo, 我們最初創(chuàng)建了兩個(gè)線程組:

EventLoopGroup bossGroup = new NioEventLoopGroup(1);
EventLoopGroup workerGroup = new NioEventLoopGroup();

創(chuàng)建EventLoopGroup的構(gòu)造方法

這里, 我們跟隨創(chuàng)建EventLoopGroup的構(gòu)造方法, 來(lái)繼續(xù)學(xué)習(xí)NioEventLoopGroup的創(chuàng)建過(guò)程

以workerGroup為例我們跟進(jìn)其構(gòu)造方法:

public NioEventLoopGroup() {
    this(0);
}

繼續(xù)跟進(jìn)this(0):

public NioEventLoopGroup(int nThreads) {
    this(nThreads, (Executor) null);
}

這里的nThreads就是剛傳入的0, 繼續(xù)跟進(jìn):

public NioEventLoopGroup(int nThreads, Executor executor) {
    this(nThreads, executor, SelectorProvider.provider());
}

這里nThreads仍然為0, executor為null, 這個(gè)execute是用于開啟NioEventLoop線程所需要的線程執(zhí)行器, SelectorProvider.provider()是用于創(chuàng)建selector, 這個(gè)之后會(huì)講到

我們一直跟到構(gòu)造方法最后:

public NioEventLoopGroup(int nThreads, Executor executor, EventExecutorChooserFactory chooserFactory, 
                         final SelectorProvider selectorProvider, 
                         final SelectStrategyFactory selectStrategyFactory, 
                         final RejectedExecutionHandler rejectedExecutionHandler) {
    super(nThreads, executor, chooserFactory, selectorProvider, selectStrategyFactory, rejectedExecutionHandler);
}

這里調(diào)用了父類的構(gòu)造方法

跟進(jìn)super, 進(jìn)入了其父類MultithreadEventExecutorGroup的構(gòu)造方法中:

protected MultithreadEventLoopGroup(int nThreads, Executor executor, EventExecutorChooserFactory chooserFactory, 
                                 Object... args) {
    super(nThreads == 0 ? DEFAULT_EVENT_LOOP_THREADS : nThreads, executor, chooserFactory, args);
}

這里我們看到, 如果傳入的線程數(shù)量參數(shù)為0, 則會(huì)給一個(gè)默認(rèn)值, 這個(gè)默認(rèn)值就是兩倍的CPU核數(shù), chooserFactory是用于創(chuàng)建線程選擇器, 之后會(huì)講到, 繼續(xù)跟代碼之后, 我們就看到了創(chuàng)建NioEventLoop的真正邏輯, 在MultithreadEventExecutorGroup類的構(gòu)造方法中

跟到MultithreadEventExecutorGroup類的構(gòu)造方法

protected MultithreadEventExecutorGroup(int nThreads, Executor executor, 
                                        EventExecutorChooserFactory chooserFactory, Object... args) {
    //代碼省略
    if (executor == null) {
        //創(chuàng)建一個(gè)新的線程執(zhí)行器(1)
        executor = new ThreadPerTaskExecutor(newDefaultThreadFactory());
    }
    //構(gòu)造NioEventLoop(2)
    children = new EventExecutor[nThreads];
    for (int i = 0; i < nThreads; i ++) {
        boolean success = false;
        try {
            children[i] = newChild(executor, args);
            success = true;
        } catch (Exception e) {
            throw new IllegalStateException("failed to create a child event loop", e);
        } finally {
           //代碼省略
        }
    }
    //創(chuàng)建線程選擇器(3)
    chooser = chooserFactory.newChooser(children);
    //代碼省略
}

這邊將代碼主要分為三個(gè)步驟:

1.創(chuàng)建線程執(zhí)行器

2.創(chuàng)建EventLoop

3.創(chuàng)建線程選擇器

這一小節(jié)我們主要剖析第1步, 創(chuàng)建線程執(zhí)行器

這里有個(gè)new DefaultThreadFactory()創(chuàng)建一個(gè)DefaultThreadFactory對(duì)象, 這個(gè)對(duì)象作為參數(shù)傳入ThreadPerTaskExecutor的構(gòu)造函數(shù),  DefaultThreadFactory顧名思義, 是一個(gè)線程工廠, 用于創(chuàng)建線程的, 簡(jiǎn)單看下這個(gè)類的繼承關(guān)系:

public class DefaultThreadFactory implements ThreadFactory{//類體
}

我們繼續(xù)跟進(jìn)該類的構(gòu)造方法:

protected ThreadFactory newDefaultThreadFactory() {
    return new DefaultThreadFactory(getClass());
}

其中g(shù)etClass()就是當(dāng)前類的class對(duì)象, 而當(dāng)前類是NioEventLoopGroup

繼續(xù)跟進(jìn)到DefaultThreadFactory的構(gòu)造方法中:

public DefaultThreadFactory(Class<?> poolType) {
    this(poolType, false, Thread.NORM_PRIORITY);
}

poolType是NioEventLoop的class對(duì)象, Thread.NORM_PRIORITY是設(shè)置默認(rèn)的優(yōu)先級(jí)為5

繼續(xù)跟構(gòu)造方法:

public DefaultThreadFactory(Class<?> poolType, boolean daemon, int priority) {
    this(toPoolName(poolType), daemon, priority);
}

這里的toPoolName(poolType)是將線程組命名, 這里返回后結(jié)果是"nioEventLoopGroup"(開n頭小寫), daemon為false, priority為5

繼續(xù)跟構(gòu)造方法:

public DefaultThreadFactory(String poolName, boolean daemon, int priority) {
    this(poolName, daemon, priority, System.getSecurityManager() == null ?
            Thread.currentThread().getThreadGroup() : System.getSecurityManager().getThreadGroup());
}

 System.getSecurityManager() == null ? Thread.currentThread().getThreadGroup() : System.getSecurityManager().getThreadGroup() 

這段代碼是通過(guò)三目運(yùn)算創(chuàng)建jdk底層的線程組

繼續(xù)跟this():

public DefaultThreadFactory(String poolName, boolean daemon, int priority, ThreadGroup threadGroup) {
    //省略驗(yàn)證代碼
    //線程名字前綴
    prefix = poolName + '-' + poolId.incrementAndGet() + '-';
    this.daemon = daemon;
    //優(yōu)先級(jí)
    this.priority = priority;
    //初始化線程組
    this.threadGroup = threadGroup;
}

這里初始化了DefaultThreadFactory的屬性, prefix為poolName(也就是nioEventLoopGroup)+'-'+線程組id(原子自增)+'-'

以及初始化了優(yōu)先級(jí)和jdk底層的線程組等屬性

回到最初MultithreadEventExecutorGroup類的構(gòu)造方法中, 我們看繼續(xù)看第一步:

//創(chuàng)建一個(gè)新的線程執(zhí)行器(1)
executor = new ThreadPerTaskExecutor(newDefaultThreadFactory());

我們繼續(xù)跟進(jìn)ThreadPerTaskExecutor的類中

public final class ThreadPerTaskExecutor implements Executor {
    private final ThreadFactory threadFactory;
    public ThreadPerTaskExecutor(ThreadFactory threadFactory) {
        if (threadFactory == null) {
            throw new NullPointerException("threadFactory");
        }
        this.threadFactory = threadFactory;
    }
    @Override
    public void execute(Runnable command) {
        //起一個(gè)線程
        threadFactory.newThread(command).start();
    }
}

我們發(fā)現(xiàn)這個(gè)類非常簡(jiǎn)單, 繼承了jdk的Executor類, 從繼承關(guān)系中我就能猜想到, 而這個(gè)類就是用于開啟線程的線程執(zhí)行器

構(gòu)造方法傳入ThreadFactory類型的參數(shù), 這個(gè)ThreadFactory就是我們剛才剖析的DefaultThreadFactory, 這個(gè)類繼承了ThreadFactory, 所以在構(gòu)造方法中初始化了ThreadFactory類型的屬性

我們?cè)倏粗貙懙?nbsp;execute(Runnable command) 方法, 傳入一個(gè)任務(wù), 然后由threadFactory對(duì)象創(chuàng)建一個(gè)線程執(zhí)行該任務(wù)

這個(gè)execute(Runnable command)方法, 其實(shí)就是用開開啟NioEventLoop線程用的, 那么NioEventLoop什么時(shí)候開啟的, 后面章節(jié)會(huì)進(jìn)行剖析

這樣, 通過(guò) executor = new ThreadPerTaskExecutor(newDefaultThreadFactory()) 這種方式就返回了一個(gè)線程執(zhí)行器Executor, 用于開啟NioEventLoop線程

以上就是Netty組件NioEventLoopGroup創(chuàng)建線程執(zhí)行器源碼解析的詳細(xì)內(nèi)容,更多關(guān)于Netty NioEventLoopGroup線程執(zhí)行器的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • SpringBoot配置Https訪問(wèn)的詳細(xì)步驟

    SpringBoot配置Https訪問(wèn)的詳細(xì)步驟

    HTTP(Hypertext transfer protocal)是一種詳細(xì)規(guī)定了瀏覽器和萬(wàn)維網(wǎng)服務(wù)器之間相互通信的規(guī)則,通過(guò)因特網(wǎng)傳送萬(wàn)維網(wǎng)文檔的數(shù)據(jù)傳送協(xié)議,這篇文章主要介紹了SpringBoot配置Https訪問(wèn)的詳細(xì)步驟,需要的朋友可以參考下
    2024-02-02
  • SpringBoot集成PostgreSQL并設(shè)置最大連接數(shù)

    SpringBoot集成PostgreSQL并設(shè)置最大連接數(shù)

    本文主要介紹了SpringBoot集成PostgreSQL并設(shè)置最大連接數(shù),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-11-11
  • MyBatis分頁(yè)插件PageHelper的具體使用

    MyBatis分頁(yè)插件PageHelper的具體使用

    這篇文章主要介紹了MyBatis分頁(yè)插件PageHelper的具體使用,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2019-02-02
  • java easyUI實(shí)現(xiàn)自定義網(wǎng)格視圖實(shí)例代碼

    java easyUI實(shí)現(xiàn)自定義網(wǎng)格視圖實(shí)例代碼

    這篇文章主要給大家介紹了關(guān)于java easyUI實(shí)現(xiàn)自定義網(wǎng)格視圖的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2018-10-10
  • Java使用easyExcel導(dǎo)出數(shù)據(jù)及單元格多張圖片

    Java使用easyExcel導(dǎo)出數(shù)據(jù)及單元格多張圖片

    除了平時(shí)簡(jiǎn)單的數(shù)據(jù)導(dǎo)出需求外,我們也經(jīng)常會(huì)遇到一些有固定格式或者模板要求的數(shù)據(jù)導(dǎo)出,下面這篇文章主要給大家介紹了關(guān)于Java使用easyExcel導(dǎo)出數(shù)據(jù)及單元格多張圖片的相關(guān)資料,需要的朋友可以參考下
    2023-05-05
  • Spring Cloud Feign實(shí)例講解學(xué)習(xí)

    Spring Cloud Feign實(shí)例講解學(xué)習(xí)

    這篇文章主要介紹了Spring Cloud Feign實(shí)例講解學(xué)習(xí),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2018-02-02
  • IDEA使用properties配置文件進(jìn)行mysql數(shù)據(jù)庫(kù)連接的教程圖解

    IDEA使用properties配置文件進(jìn)行mysql數(shù)據(jù)庫(kù)連接的教程圖解

    Properties類是 鍵和值均為字符串的可以永久存儲(chǔ)到文件中的key-value集合。這篇文章主要介紹了IDEA使用properties配置文件進(jìn)行mysql數(shù)據(jù)路連接 ,需要的朋友可以參考下
    2018-10-10
  • springcloud如何使用Feign后臺(tái)內(nèi)部傳遞MultipartFile

    springcloud如何使用Feign后臺(tái)內(nèi)部傳遞MultipartFile

    這篇文章主要介紹了springcloud如何使用Feign后臺(tái)內(nèi)部傳遞MultipartFile,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-03-03
  • java反射獲取和調(diào)用方法

    java反射獲取和調(diào)用方法

    本篇內(nèi)容主要給大家詳細(xì)講解了java反射獲取方法以及調(diào)用方法,需要的朋友參考學(xué)習(xí)一下吧。
    2017-12-12
  • SpringBoot中的@Conditional?注解的使用

    SpringBoot中的@Conditional?注解的使用

    @Conditional是Spring4新提供的注解,它的作用是按照一定的條件進(jìn)行判斷,滿足條件的才給容器注冊(cè)Bean,本文主要介紹了SpringBoot中的@Conditional?注解的使用
    2024-01-01

最新評(píng)論