环球关注:【Netty源码分析】02 Netty Server 启动流程 下

2023-03-28 12:17:28 来源:腾讯云

上面我们分析了initAndRegister()方法的核心流程,Channel准备工作基本也都完成了:

ChannelNioEventLoop进行了关联;Channel也注册到Selector上了;NioEventLoop线程也启动完成,开始轮询事件、处理事件。

这里还遗漏了两个事情:channel和端口绑定以及channelSelector注册OP_ACCEPT。这就是在doBind()方法中另一个重要的方法:doBind0()中进行完成的。


(资料图片)

doBind0

if (regFuture.isDone()) {    ChannelPromise promise = channel.newPromise();    doBind0(regFuture, channel, localAddress, promise);    return promise;} else {    final PendingRegistrationPromise promise = new PendingRegistrationPromise(channel);    //register还未完成,则添加listener,待注册完成再执行doBind0()进行server端口绑定    regFuture.addListener(new ChannelFutureListener() {        @Override        public void operationComplete(ChannelFuture future) throws Exception {            Throwable cause = future.cause();            if (cause != null) {                promise.setFailure(cause);            } else {                promise.registered();                doBind0(regFuture, channel, localAddress, promise);            }        }    });    return promise;}

上面代码一大堆,核心就是调用doBind0()方法,但是执行该方法前必须保证上一步initAndRegister()方法中执行完成。通过regFuture.isDone()进行判断,具体设置位置见下:

//AbstractChannel.AbstractUnsafe#register0pipeline.invokeHandlerAddedIfNeeded();// 将指定的promise标记为成功:regFuture.isDone()=true,doBind0()才能开始执行safeSetSuccess(promise);pipeline.fireChannelRegistered();

进行向下跟踪,来到了如下代码处,会发现需要调用channel.bind()方法,但是不是在当前线程中直接调用,而是封装成task放入到NioEventLoop的任务队列taskQueue中,由NioEventLoop线程执行:

private static void doBind0(        final ChannelFuture regFuture, final Channel channel,        final SocketAddress localAddress, final ChannelPromise promise) {    channel.eventLoop().execute(new Runnable() {        @Override        public void run() {            if (regFuture.isSuccess()) {                channel.bind(localAddress, promise).addListener(ChannelFutureListener.CLOSE_ON_FAILURE);            } else {                promise.setFailure(regFuture.cause());            }        }   });}

这时的NioEventLoop线程是已经启动并开始工作的,所以channel.bind()这里是可以执行的。

层层调用最终是在pipeline中的head这个节点进行处理的:

public final void bind(final SocketAddress localAddress, final ChannelPromise promise) {    assertEventLoop();    if (!promise.setUncancellable() || !ensureOpen(promise)) {        return;    }    //还没有绑定端口,isActive()返回false    boolean wasActive = isActive();    try {        //调用底层java api,将channel绑定到具体端口上        doBind(localAddress);    } catch (Throwable t) {        safeSetFailure(promise, t);        closeIfClosed();        return;    }    //经过上面绑定端口,这时isActive()=true    if (!wasActive && isActive()) {        invokeLater(new Runnable() {            @Override            public void run() {                //触发server handler的channelActive()方法                pipeline.fireChannelActive();            }        });    }    safeSetSuccess(promise);}

这个方法主要完成2件事:

doBind():调用java api,将channel绑定到具体端口上;pipeline.fireChannelActive():将pipeline.fireChannelActive()放入到NioEventLoop线程中执行;

下面我们再来看下pipeline.fireChannelActive()

public void channelActive(ChannelHandlerContext ctx) { ctx.fireChannelActive(); readIfIsAutoRead();}

该方法主要做2件事:

ctx.fireChannelActive():触发handler#channelActive()调用,表示当前channel已处于激活状态,可以正常工作了;readIfIsAutoRead():从名称看就是,如果配置autoRead,调用readIfIsAutoRead()直接进行read操作;readIfIsAutoRead()会调用tail.read(),然后一层层往前查找,最终调用的是head#read()方法。
protected void doBeginRead() throws Exception {    final SelectionKey selectionKey = this.selectionKey;    if (!selectionKey.isValid()) {        return;    }    readPending = true;    final int interestOps = selectionKey.interestOps();    // 将SelectionKey当前的操作位与注册操作位进行按位与操作,如果等于0,说明目前并没有设置注册操作位    if ((interestOps & readInterestOp) == 0) {        // Server Channel会在这里注册真正的ACCEPT事件        selectionKey.interestOps(interestOps | readInterestOp);    }}

channel绑定好端口后,触发了channelActive()方法回调,channel真正进入可以正常工作状态,这时还差最后一步:注册OP_ACCEPT事件。

总结

这样,Netty整体启动就全部完成,NioServerSocketChannel这时就可以正常接收到客户端连接请求。

标签

汤臣倍健2021年市占率10.3% 稳居行业第一

VDS行业发展空间较大、集中度较低。国内膳食营养补充剂(VDS)发展历史尚短,居民的消费意识和习惯尚未完...

2022-05-22 21:06:33

郴州安仁文旅项目集中开工 总投资1000万元

3月16日,安仁县举行文旅项目集中开工活动,县委书记王洪灿在开工仪式上宣布:湘南起义旧址群——朱毛井...

2022-03-20 15:40:46

2022年郴州计划重点推进文旅项目101个 总投资354亿元

3月16日,我市举行全市文旅项目和城市大提质大融城项目集中开工仪式,市委书记吴巨培宣布项目开工。郴州...

2022-03-20 15:39:41

宿州泗县深入推进文旅融合发展 擦亮城市品牌

近年来,泗县以争创安徽省文化旅游名县为目标,深入推进文旅融合发展,努力擦亮水韵泗州 运河名城城市...

2022-03-20 15:38:59

汽车零部件产业“领头羊” 锦州力争一季度“开门红”

3月16日,记者从锦州汽车零部件产业的领头羊——锦州万得集团获悉,今年前两个月,企业订单充足,正铆足...

2022-03-20 15:37:41

油价或有望冲击“九元”大关 宁波新能源汽车市场如何

新一轮国内成品油调价窗口于3月17日24时开启,油价或有望冲击九元大关。前一天晚上11点,鄞州区不少加油...

2022-03-20 15:34:38

从水塘到“云”端 全国最大高邮鸭养殖基地实现智慧养殖

随着新一代数字技术的蓬勃发展,以新兴技术推动现代化新农村建设正成为助力乡村振兴的重要手段。1个人能...

2022-03-20 15:33:17

淡季不忘引流 京郊民宿市场有望迎来回暖

旅游淡季中的京郊民宿有望成为市场中最先复苏的板块。3月17日,北京商报记者调查发现,虽然正值旅游淡季...

2022-03-20 15:32:01

镇江乡村一二三产业融合发展 闯出“镇江之路”

从烹饪江鲜河豚的个体小饭店到规模化的江岛乡村旅游产业集群,从白兔草莓丁庄葡萄的单个农户种植到茅山...

2022-03-20 15:31:11

总投资30亿元 盐城东台8个重大产业项目相继开工

总投资30亿元的精密电子元器件项目、同益电子项目,总投资10亿元的金利美精密组件项目、天永智能设备项...

2022-03-20 15:30:13
x 广告
x 广告

Copyright  2015-2022 海峡粮油网版权所有  备案号:皖ICP备2022009963号-10   联系邮箱:396 029 142 @qq.com