Netty服务端的业务流程分析
Netty的服务端怎么和java NIO联系起来的,一直很好奇这块内容,这里跟下代码,下篇文章看下Channel相关的知识。
finalChannelFuture initAndRegister(){finalChannel channel = channelFactory().newChannel();//try{init(channel);}catch(Throwable t){channel.unsafe().closeForcibly();//立即关闭通道且不会触发事件//因为这个通道还没有注册到EventLoop,所以我们需要强制GlobalEventExecutor的使用。returnnewDefaultChannelPromise(channel,GlobalEventExecutor.INSTANCE).setFailure(t);}//注册一个EventLoopChannelFuture regFuture = group().register(channel);//注册失败if(regFuture.cause()!=null){if(channel.isRegistered()){channel.close();}else{channel.unsafe().closeForcibly();}}// If we are here and the promise is not failed, it's one of the following cases:// 程序运行到这里且promise没有失败,可能有如下几种情况// 1) If we attempted registration from the event loop, the registration has been completed at this point.// 如果试图注册到一个EventLoop,该注册完成,// i.e. It's safe to attempt bind() or connect() now because the channel has been registered.// 2) If we attempted registration from the other thread, the registration request has been successfully// added to the event loop's task queue for later execution.// 如果试图注册到其他线程,该注册已经成功,但是没有完成,添加一个事件到任务队列中,等会执行// i.e. It's safe to attempt bind() or connect() now:// because bind() or connect() will be executed *after* the scheduled registration task is executed// because register(), bind(), and connect() are all bound to the same thread.return regFuture;}
@OverridepublicChannelFutureregister(Channel channel){return next().register(channel);}
@OverridepublicChannelFutureregister(finalChannel channel,finalChannelPromise promise){if(channel ==null){thrownewNullPointerException("channel");}if(promise ==null){thrownewNullPointerException("promise");}channel.unsafe().register(this, promise);return promise;}
@Overridepublicfinalvoidregister(EventLoop eventLoop,finalChannelPromise promise){if(eventLoop ==null){thrownewNullPointerException("eventLoop");}if(isRegistered()){promise.setFailure(newIllegalStateException("registered to an event loop already"));return;}if(!isCompatible(eventLoop)){promise.setFailure(newIllegalStateException("incompatible event loop type: "+ eventLoop.getClass().getName()));return;}AbstractChannel.this.eventLoop = eventLoop;if(eventLoop.inEventLoop()){register0(promise);}else{try{eventLoop.execute(newOneTimeTask(){@Overridepublicvoid run(){register0(promise);}});}catch(Throwable t){logger.warn("Force-closing a channel whose registration task was not accepted by an event loop: {}",AbstractChannel.this, t);closeForcibly();closeFuture.setClosed();safeSetFailure(promise, t);}}}
@Overrideprotectedvoid doRegister()throwsException{boolean selected =false;for(;;){try{selectionKey = javaChannel().register(eventLoop().selector,0,this);return;}catch(CancelledKeyException e){if(!selected){// Force the Selector to select now as the "canceled" SelectionKey may still be// cached and not removed because no Select.select(..) operation was called yet.eventLoop().selectNow();selected =true;}else{// We forced a select operation on the selector before but the SelectionKey is still cached// for whatever reason. JDK bug ?throw e;}}}}
wakeup 方法所得的结果privatestaticvoid doBind0(finalChannelFuture regFuture,finalChannel channel,finalSocketAddress localAddress,finalChannelPromise promise){// This method is invoked before channelRegistered() is triggered. Give user handlers a chance to set up// the pipeline in its channelRegistered() implementation.channel.eventLoop().execute(newRunnable(){@Overridepublicvoid run(){if(regFuture.isSuccess()){channel.bind(localAddress, promise).addListener(ChannelFutureListener.CLOSE_ON_FAILURE);}else{promise.setFailure(regFuture.cause());}}});}
@Overrideprotectedvoid run(){for(;;){boolean oldWakenUp = wakenUp.getAndSet(false);try{if(hasTasks()){selectNow();}else{select(oldWakenUp);// 'wakenUp.compareAndSet(false, true)' is always evaluated// before calling 'selector.wakeup()' to reduce the wake-up// overhead. (Selector.wakeup() is an expensive operation.)//// However, there is a race condition in this approach.// The race condition is triggered when 'wakenUp' is set to// true too early.//// 'wakenUp' is set to true too early if:// 1) Selector is waken up between 'wakenUp.set(false)' and// 'selector.select(...)'. (BAD)// 2) Selector is waken up between 'selector.select(...)' and// 'if (wakenUp.get()) { ... }'. (OK)//// In the first case, 'wakenUp' is set to true and the// following 'selector.select(...)' will wake up immediately.// Until 'wakenUp' is set to false again in the next round,// 'wakenUp.compareAndSet(false, true)' will fail, and therefore// any attempt to wake up the Selector will fail, too, causing// the following 'selector.select(...)' call to block// unnecessarily.//// To fix this problem, we wake up the selector again if wakenUp// is true immediately after selector.select(...).// It is inefficient in that it wakes up the selector for both// the first case (BAD - wake-up required) and the second case// (OK - no wake-up required).if(wakenUp.get()){selector.wakeup();}}cancelledKeys =0;needsToSelectAgain =false;finalint ioRatio =this.ioRatio;if(ioRatio ==100){processSelectedKeys();runAllTasks();}else{finallong ioStartTime =System.nanoTime();processSelectedKeys();finallong ioTime =System.nanoTime()- ioStartTime;runAllTasks(ioTime *(100- ioRatio)/ ioRatio);}if(isShuttingDown()){closeAll();if(confirmShutdown()){break;}}}catch(Throwable t){logger.warn("Unexpected exception in the selector loop.", t);// Prevent possible consecutive immediate failures that lead to// excessive CPU consumption.try{Thread.sleep(1000);}catch(InterruptedException e){// Ignore.}}}}
Netty服务端的业务流程分析的更多相关文章
- Netty服务端的启动源码分析
		
ServerBootstrap的构造: public class ServerBootstrap extends AbstractBootstrap<ServerBootstrap, Serve ...
 - Netty之旅三:Netty服务端启动源码分析,一梭子带走!
		
Netty服务端启动流程源码分析 前记 哈喽,自从上篇<Netty之旅二:口口相传的高性能Netty到底是什么?>后,迟迟两周才开启今天的Netty源码系列.源码分析的第一篇文章,下一篇我 ...
 - 【Netty源码分析】Netty服务端bind端口过程
		
这一篇博客我们介绍一下Netty服务端绑定端口的过程,我们通过跟踪代码一直到NIO原生绑定端口的操作. 绑定端口操作 ChannelFuture future = serverBootstrap.bi ...
 - Netty 服务端启动过程
		
在 Netty 中创建 1 个 NioServerSocketChannel 在指定的端口监听客户端连接,这个过程主要有以下 个步骤: 创建 NioServerSocketChannel 初始化并注 ...
 - Netty 服务端创建
		
参考:http://blog.csdn.net/suifeng3051/article/details/28861883?utm_source=tuicool&utm_medium=refer ...
 - netty服务端启动--ServerBootstrap源码解析
		
netty服务端启动--ServerBootstrap源码解析 前面的第一篇文章中,我以spark中的netty客户端的创建为切入点,分析了netty的客户端引导类Bootstrap的参数设置以及启动 ...
 - Netty服务端NioEventLoop启动及新连接接入处理
		
一 Netty服务端NioEventLoop的启动 Netty服务端创建.初始化完成后,再向Selector上注册时,会将服务端Channel与NioEventLoop绑定,绑定之后,一方面会将服务端 ...
 - Netty服务端Channel的创建与初始化
		
Netty创建服务端Channel时,从服务端 ServerBootstrap 类的 bind 方法进入,下图是创建服务端Channel的函数调用链.在后续代码中通过反射的方式创建服务端Channel ...
 - netty服务端客户端启动流程分析
		
服务端启动流程 我们回顾前面讲解的netty启动流程,服务端这边有两个EventLoopGroup,一个专门用来处理连接,一个用来处理后续的io事件 服务端启动还是跟nio一样,绑定端口进行监听,我们 ...
 
随机推荐
- BZOJ - 2142 礼物 (扩展Lucas定理)
			
扩展Lucas定理模板题(貌似这玩意也只能出模板题了吧~~本菜鸡见识鄙薄,有待指正) 原理: https://blog.csdn.net/hqddm1253679098/article/details ...
 - LeetCode 336. Palindrome Pairs
			
原题链接在这里:https://leetcode.com/problems/palindrome-pairs/ 题目: Given a list of unique words, find all p ...
 - Linux基础命令-系统时间
			
Linux启动时从硬件读取日期和时间信息,读取完成以后,就不再与硬件相关联 Linux的两种时钟 系统时钟:由Linux内核通过CPU的工作频率进行的: date:显示系统时间 +%D +%F dat ...
 - Spring MVC 项目示例
			
Spring MVC是Spring Framework的一部分,是基于Java实现MVC的轻量级Web框架.Spring的web框架围绕DispatcherServlet设计, 作用是将请求分发到不同 ...
 - 机器学习:评价分类结果(Precision - Recall 的平衡、P - R 曲线)
			
一.Precision - Recall 的平衡 1)基础理论 调整阈值的大小,可以调节精准率和召回率的比重: 阈值:threshold,分类边界值,score > threshold 时分类为 ...
 - du 命令 -目前的目录所占的磁盘空间
			
Linux du命令也是查看使用空间的,但是与df命令不同的是Linux du命令是对文件和目录磁盘使用的空间的查看,还是和df命令有一些区别的. 1.命令格式: du [选项][文件] 2.命令功能 ...
 - XXXAction-validation.xml文件中报错:Referenced file Contains errors
			
我们需要引用与验证器配置相关的dtd文件,这个文件可以在xwork-core-2.3.1.2.jar下找到(xwork-validator-1.0.3.dtd) 网上有很多处理办法,如下所示: 1.直 ...
 - Mongodb 3.6 副本集测试及添加删除节点等操作
			
下载tar包并安装curl -O https://fastdl.mongodb.org/linux/mongodb-linux-x86_64-rhel70-3.6.8.tgz [root@mysqlt ...
 - Maven的Snapshot版本与Release版本
			
1. Snapshot版本代表不稳定.尚处于开发中的版本 2. Release版本则代表稳定的版本 3. 什么情况下该用SNAPSHOT? 协同开发时,如果A依赖构件B,由于B会更新,B应该使用SNA ...
 - String是基本的数据类型吗?
			
String不是基本的数据类型,是final修饰的java类,java中的基本类型一共有8个,它们分别为: 1 字符类型:byte,char 2 基本整型:short,int,long 3 浮点型:f ...