netty筆記-NioEventLoopGroup

1.EventExecutorGroup

從命名來看蜈垮,其是EventExecutor的一個容器,next方法就是選擇一個EventExecutor

public interface EventExecutorGroup extends ScheduledExecutorService, Iterable<EventExecutor> {
    EventExecutor next();
}

2.EventExecutor

同時EventExecutor也繼承了EventExecutorGroup址晕,這點有些不可理解,既然是一個group,那么其next方法如何實現(xiàn)?

3.next方法實現(xiàn)

3.1 MultithreadEventExecutorGroup

MultithreadEventExecutorGroup實現(xiàn)了next方法,其使用了一個EventExecutorChooser選擇器來選擇EventExecutor

public abstract class MultithreadEventExecutorGroup extends AbstractEventExecutorGroup {

    private final EventExecutorChooserFactory.EventExecutorChooser chooser;

    @Override
    public EventExecutor next() {
        return chooser.next();
    }
}

3.2 AbstractEventExecutor

AbstractEventExecutor直接返回的是this,所以其是一個單容器實現(xiàn)

public abstract class AbstractEventExecutor extends AbstractExecutorService implements EventExecutor {
    @Override
    public EventExecutor next() {
        return this;
    }

3.3 AbstractEventExecutorGroup

基本可以認為是EventExecutor的代理執(zhí)行,代碼基本類似

public abstract class AbstractEventExecutorGroup implements EventExecutorGroup {
@Override
public Future<?> submit(Runnable task) {
return next().submit(task);
}
}

4.EventExecutorChooser

public final class DefaultEventExecutorChooserFactory implements EventExecutorChooserFactory {

    public static final DefaultEventExecutorChooserFactory INSTANCE = new DefaultEventExecutorChooserFactory();

    private DefaultEventExecutorChooserFactory() { }

    @SuppressWarnings("unchecked")
    @Override
    public EventExecutorChooser newChooser(EventExecutor[] executors) {
        if (isPowerOfTwo(executors.length)) {
            return new PowerOfTwoEventExecutorChooser(executors);
        } else {
            return new GenericEventExecutorChooser(executors);
        }
    }

    private static boolean isPowerOfTwo(int val) {
        return (val & -val) == val;
    }

    private static final class PowerOfTwoEventExecutorChooser implements EventExecutorChooser {
        private final AtomicInteger idx = new AtomicInteger();
        private final EventExecutor[] executors;

        PowerOfTwoEventExecutorChooser(EventExecutor[] executors) {
            this.executors = executors;
        }

        @Override
        public EventExecutor next() {
            return executors[idx.getAndIncrement() & executors.length - 1];
        }
    }

    private static final class GenericEventExecutorChooser implements EventExecutorChooser {
        private final AtomicInteger idx = new AtomicInteger();
        private final EventExecutor[] executors;

        GenericEventExecutorChooser(EventExecutor[] executors) {
            this.executors = executors;
        }

        @Override
        public EventExecutor next() {
            return executors[Math.abs(idx.getAndIncrement() % executors.length)];
        }
    }
}

線程池數(shù)量使用2的冪次方逛漫,這樣線程池選擇線程時使用位操作赌莺,能使性能最高震叙。

5.NioEventLoop的初始化

在MultithreadEventExecutorGroup的構(gòu)造函數(shù)中栈雳,通過newChild方法初始化

NioEventLoopGroup重寫了newChild方法怯邪,返回的是NioEventLoop對象

   protected MultithreadEventExecutorGroup(int nThreads, Executor executor,
                                            EventExecutorChooserFactory chooserFactory, Object... args) {
        if (nThreads <= 0) {
            throw new IllegalArgumentException(String.format("nThreads: %d (expected: > 0)", nThreads));
        }

        if (executor == null) {
            executor = new ThreadPerTaskExecutor(newDefaultThreadFactory());
        }

        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) {
                // TODO: Think about if this is a good exception type
                throw new IllegalStateException("failed to create a child event loop", e);
            } finally {
                if (!success) {
                    for (int j = 0; j < i; j ++) {
                        children[j].shutdownGracefully();
                    }

                    for (int j = 0; j < i; j ++) {
                        EventExecutor e = children[j];
                        try {
                            while (!e.isTerminated()) {
                                e.awaitTermination(Integer.MAX_VALUE, TimeUnit.SECONDS);
                            }
                        } catch (InterruptedException interrupted) {
                            // Let the caller handle the interruption.
                            Thread.currentThread().interrupt();
                            break;
                        }
                    }
                }
            }
        }


public class NioEventLoopGroup extends MultithreadEventLoopGroup {

    @Override
    protected EventLoop newChild(Executor executor, Object... args) throws Exception {
        return new NioEventLoop(this, executor, (SelectorProvider) args[0],
            ((SelectStrategyFactory) args[1]).newSelectStrategy(), (RejectedExecutionHandler) args[2]);
    }
}

最終的初始化結(jié)果:

以上分析鸠踪,基本可以看到EventExecutorGroup的核心在于選擇EventExecutor,具體的執(zhí)行方法都在EventExecutor中實現(xiàn)

參考:
自頂向下深入分析Netty(四)--EventLoop-1

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末丙者,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子营密,更是在濱河造成了極大的恐慌械媒,老刑警劉巖,帶你破解...
    沈念sama閱讀 212,383評論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異纷捞,居然都是意外死亡痢虹,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,522評論 3 385
  • 文/潘曉璐 我一進店門主儡,熙熙樓的掌柜王于貴愁眉苦臉地迎上來奖唯,“玉大人,你說我怎么就攤上這事糜值》峤荩” “怎么了?”我有些...
    開封第一講書人閱讀 157,852評論 0 348
  • 文/不壞的土叔 我叫張陵寂汇,是天一觀的道長病往。 經(jīng)常有香客問我,道長骄瓣,這世上最難降的妖魔是什么停巷? 我笑而不...
    開封第一講書人閱讀 56,621評論 1 284
  • 正文 為了忘掉前任,我火速辦了婚禮累贤,結(jié)果婚禮上叠穆,老公的妹妹穿的比我還像新娘少漆。我一直安慰自己臼膏,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 65,741評論 6 386
  • 文/花漫 我一把揭開白布示损。 她就那樣靜靜地躺著渗磅,像睡著了一般。 火紅的嫁衣襯著肌膚如雪检访。 梳的紋絲不亂的頭發(fā)上始鱼,一...
    開封第一講書人閱讀 49,929評論 1 290
  • 那天,我揣著相機與錄音脆贵,去河邊找鬼医清。 笑死,一個胖子當(dāng)著我的面吹牛卖氨,可吹牛的內(nèi)容都是我干的会烙。 我是一名探鬼主播,決...
    沈念sama閱讀 39,076評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼筒捺,長吁一口氣:“原來是場噩夢啊……” “哼柏腻!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起系吭,我...
    開封第一講書人閱讀 37,803評論 0 268
  • 序言:老撾萬榮一對情侶失蹤五嫂,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體沃缘,經(jīng)...
    沈念sama閱讀 44,265評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡躯枢,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,582評論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了槐臀。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片闺金。...
    茶點故事閱讀 38,716評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖峰档,靈堂內(nèi)的尸體忽然破棺而出败匹,到底是詐尸還是另有隱情,我是刑警寧澤讥巡,帶...
    沈念sama閱讀 34,395評論 4 333
  • 正文 年R本政府宣布掀亩,位于F島的核電站,受9級特大地震影響欢顷,放射性物質(zhì)發(fā)生泄漏槽棍。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 40,039評論 3 316
  • 文/蒙蒙 一抬驴、第九天 我趴在偏房一處隱蔽的房頂上張望炼七。 院中可真熱鬧,春花似錦布持、人聲如沸豌拙。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,798評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽按傅。三九已至,卻和暖如春胧卤,著一層夾襖步出監(jiān)牢的瞬間唯绍,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,027評論 1 266
  • 我被黑心中介騙來泰國打工枝誊, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留况芒,地道東北人。 一個月前我還...
    沈念sama閱讀 46,488評論 2 361
  • 正文 我出身青樓叶撒,卻偏偏與公主長得像绝骚,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子痊乾,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,612評論 2 350

推薦閱讀更多精彩內(nèi)容