Netty中的策略者模式


策略者模式的特點

在設計類的繼承體系時,我們會刻意的把公共的部分都提取到基類中

比如先設計Person類,把人類都具有的行為放到這個Person,特有的行為設計成抽象方法,讓子類具體去實現, 這樣后續無論我們再去構造學生,還是構造老師,大家都繼承Person,就達到了代碼復用的目的

但是這樣問題就來了,對老師類來說,需要有教學的行為,假如這個方法以抽象方法的形式放在基類,那么對於繼承了Person的學生類來說就不對了,因為沒有要求學生一定會教學,但是現在學生就得實現這個方法

如果我們把老師的教學的行為作為 老師類的私有, 這時候,小明教小李學習, 就意味着對小明來說,他需要教學的行為, 前前后后看起來就開始矛盾了, 到底怎么處理呢?

策略者模式,就解決了這個問題, 它把行為抽象成了接口,以接口+實現的方式,解決上面的問題, 就上面的例子來說,可以把教學設計成接口,任何類,只要實現了這個接口,就可以教學,而不一定強制要求只有老師才可以實現它

總的來說,策略模式,就是將行為抽象成接口+實現的模式

Netty中策略者模式的使用

netty的bossgroup中接收到了新的連接之后會使用選擇器Chooser,從WorkerGroup中選擇出一個EventLoop, 然后把這個連接注冊進選出的 EventLoop

netty的選擇器,使用的就是策略者模式,將選擇的行為 設計成接口,不同的選擇器根據自己不同的需求用不用的方式實現選擇器接口

行為接口

@UnstableApi
public interface EventExecutorChooserFactory {

EventExecutorChooser newChooser(EventExecutor[] executors);
@UnstableApi
interface EventExecutorChooser {
    EventExecutor next();
}
}

選擇器不同的實現:

if (isPowerOfTwo(executors.length)) {// todo 如果是2的指數倍, 返回PowerOfTwoEventExecutorChooser
    return new PowerOfTwoEventExecutorChooser(executors);
} else {// todo  否則返回同樣的實例
    return new GenericEventExecutorChooser(executors);
}

根據線程執行器的數量確定使用那種具體的行為

行為1:PowerOfTwoEventExecutorChooser

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];
    }

主要看它的executors[idx.getAndIncrement() & executors.length - 1]

進行速度更快的與運算

1 & 1 = 1
1 & 0 = 0
0 & 1 = 0

當數組的長度是2的冪次方時, 用二進制表示就是1111... 全是1, 再減去1 ,就是0111...

無論前面的數是誰,對一個 0111... 進行與運算,得到的結果就是從0-0111...大小的數, 循環往復

行為2:GenericEventExecutorChooser

private final AtomicInteger idx = new AtomicInteger();
private final EventExecutor[] executors;

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

@Override
public EventExecutor next() {
    // todo 從0開始到最后一個, 再從零開始,到最后一個
    return executors[Math.abs(idx.getAndIncrement() % executors.length)];
}

主要的一步就是Math.abs(idx.getAndIncrement() % executors.length)
可以看到,從0開始一直往后對數組的長度取余數,小數對大數取余數=小數, 保證了數組的下標從0開始遞增, 自己對自己取余數=0,保證了最大值是 數組的長度減一, 如此往復


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM