来源:腾讯云
上面我们分析了initAndRegister()
方法的核心流程,Channel
准备工作基本也都完成了:
Channel
和NioEventLoop
进行了关联;Channel
也注册到Selector
上了;NioEventLoop
线程也启动完成,开始轮询事件、处理事件。这里还遗漏了两个事情:channel
和端口绑定以及channel
向Selector
注册OP_ACCEPT
。这就是在doBind()
方法中另一个重要的方法: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
这时就可以正常接收到客户端连接请求。
关键词:
还在苦苦寻找优秀经典的名言吗?为大家整理的关于法律的名言警句
关于法律的名言警句(精选220句)在日常学习、工作或生活中,大家都有令自己印象深刻的名言吧,巧用名言有助于我们正确对待学习、生活、成长
怎样写方案才更能起到其作用呢?整理的项目合作实施方案
项目合作实施方案5篇为保证事情或工作高起点、高质量、高水平开展,往往需要预先进行方案制定工作,方案指的是为某一次行动所制定的计划类
楚辞影响最大的作品 《离骚》全文对照翻译
《离骚》全文对照翻译《离骚》是屈原的代表作,创作于楚怀王时期屈原遭谗被疏之时,是楚辞影响最大的作品。下面是《离骚》全文对照翻译...
写申请书时理由总是不够充分?为大家整理的退学申请书
退学申请书(精选12篇)在一步步向前发展的社会中,申请书在现实生活中使用广泛,申请书不同于其他书信,是一种专用书信。写申请书时理由总是
都有哪些类型的话语呢?为大家收集的感恩老师的话精选150句
感恩老师的话在生活、工作和学习中,越来越多人喜欢发表话语,话语是特定社会语境中人与人之间从事沟通的具体言语行为。那么都有哪些类...
关于我们 加入我们 联系我们 商务合作 粤ICP备18023326号-32
中国创氪网 www.chuanganwang.cn 版权所有
投稿投诉联系邮箱:85 572 98@qq.com