Netty源码剖析-发送数据
参考文献:极客时间傅健老师的《Netty源码剖析与实战》Talk is cheap.show me the code!
开始之前先介绍下Netty写数据的三种方式:
①:write:写到一个buffer,flush:把buffer里的数据发送出去
②:writeAndFlush:写到buffer,立马发送
③:write和flush之间有个ChannelOutboundBuffer
可以用生活中快递场景来类比下:
write相当于揽收到仓库,flush相当于从仓库发货,writeAndFlush相当于揽收到仓库立马就发货(类似加急件),ChannelOutboundBuffer相当于揽收和发货之间有个缓冲的仓库
----主线:如图:
----2源码演示:
在channelRead()加入断点,启动服务端和客户端
开始跟进ctx.write();一直跟下回来到:
然后继续跟进invokeWrte();
然后跟进invokeWrite0();
再继续跟进write(),一步一步则会进入到这:
上图代码执行的地方就相当于发快递的仓库这一步骤;继续往下走则能看到:
可以简单进去看一看:
它是添加到队尾;接着继续下边的“incrementPendingOutboundBytes()”;
判断待发送的数据的size是否高于高水位线,如果高于则修改为不可写状态。然后跳出来,write()完成之后开始发送了:在发送的地方加一个断点:
套路一样:
然后找到flush()
再继续跟进:
addFlush()就表示要装车发货了,跟进看看:
它的意思就是把unflushedEntry里面的数据转Flush里面去。然后跳回去:
进入flush0();
进入doWrite();
挨个走就是调用jdk的实现
----3总结:
写的本质:
-single write:sun.nio.ch.SocketChannelImpl#write(java.nio.ByteBuffer)(单个写)
-gathering write:sun.nio.ch.SocketChannelImpl#write(java.nio.ByteBuffer[],int,int)(批量写)
写数据写不进去时,会停止写,注册一个OP_WRITE事件,来通知什么时候可以写进去
OP_WRITE不是说有数据可写,而是说可以写进去,所以正常情况,不能注册,否则一直触发
批量写数据时,如果尝试写的都写进去了,接下来会尝试写更多(maxBytesPerGatheringWrite)。
只要有数据要写,且能写,则一直尝试,直到16次(writeSpinCount),写16次还没有写完,就直接schedule一个task来继续写,而不是用注册写事件来触发,更简洁有力。
待写数据太多,超过一定的水位线(writeBufferWaterMark.high()),会将可写的标志位改成false,让应用端自己做决定要不要继续写。
channelHandlerContext.channel().write():从TailContext开始执行;
channelHandlerContext.write():从当前的Context开始。
我只想做的更好,仅此而已。
Netty源码剖析-发送数据的更多相关文章
- Netty源码剖析-接受数据
参考文献:极客时间傅健老师的<Netty源码剖析与实战>Talk is cheap.show me the code! ----主线:worker thread ①多路复用器(Select ...
- Netty 源码剖析之 unSafe.write 方法
前言 在 Netty 源码剖析之 unSafe.read 方法 一文中,我们研究了 read 方法的实现,这是读取内容到容器,再看看 Netty 是如何将内容从容器输出 Channel 的吧. 1. ...
- Netty源码剖析-断开连接
参考文献:极客时间傅健老师的<Netty源码剖析与实战>Talk is cheap.show me the code! ----主线: ----源码: 在NioEventLoop的unsa ...
- Netty源码剖析-业务处理
参考文献:极客时间傅健老师的<Netty源码剖析与实战>Talk is cheap.show me the code! ----主线:worker thread 触发pipeline.fi ...
- Netty源码剖析-启动服务
参考文献:极客时间傅健老师的<Netty源码剖析与实战>Talk is cheap.show me the code! --1主线分两步: 一:首先在our thread里,如果写在mai ...
- Netty源码剖析-关闭服务
参考文献:极客时间傅健老师的<Netty源码剖析与实战>Talk is cheap.show me the code! ----主线: ----源码: 先在服务端加个断点和修改下代码:如 ...
- Netty源码剖析-构建链接
参考文献:极客时间傅健老师的<Netty源码剖析与实战>Talk is cheap.show me the code! ----主线: 和启动一样也是有两个线程完成的,boss threa ...
- Netty学习笔记(三)——netty源码剖析
1.Netty启动源码剖析 启动类: public class NettyNioServer { public static void main(String[] args) throws Excep ...
- Netty 源码剖析之 unSafe.read 方法
目录: NioSocketChannel$NioSocketChannelUnsafe 的 read 方法 首先看 ByteBufAllocator 再看 RecvByteBufAllocator.H ...
随机推荐
- MySQL数据分析-(11)表补充:数据类型
大家好,我是jacky,很高兴继续跟大家学习<Mysql 数据分析实战系列教程>,上次课程jacky讲解了表层面的增删改查,jacky说最重要的是增,增就是创建表,作为一个严谨的MySQL ...
- Contest Hunter 3101
题目 Contest Hunter 3101 阶乘分解 原题传送门 题目分析 这里介绍一个本蒟蒻自己\(yy\)出来的方法. 我们发现,对于某一个单个的整数\(n\),若\(n\)能被某一个数\(x\ ...
- 表单 Flask-WTF - 校验器
1 wtforms内置的校验器 Class wtforms.validators.DataRequired(message=None)此验证器将会检测field是否输入了数值,实际上是进行了if fi ...
- 【转】CAtlRegExp class .
CAtlRegExp Class CAtlRegExp 类用于表示并处理正则表达式.模板类,定义如下: template < class CharTraits = CAtlRECha ...
- 浏览器环境下的microtaks和macrotasks
带有可视代码执行顺序的原文链接https://jakearchibald.com/201...,此篇文字并非其完整翻译,加入了一部分自己的理解,比如将其中的task替换为macrotask或是删除了可 ...
- [Oracle]察看一张表的约束 和 察看一张表的索引
--察看一张表的约束select table_name,constraint_name,constraint_type from user_constraints where table_name=u ...
- shell 變數
echo $? 上个命令的退出状态,或函数的返回值. ref: http://c.biancheng.net/cpp/view/2739.html
- Linux嵌入式学习过程(转载)
嵌入式专业是一门实践性非常强的学科,只有多动手,多实践,多编程,多调试,多看书,多思考才能真正掌握好嵌入式开发技术.那么,如何从零开始学习嵌入式开发技术, 进入嵌入式开发大门呢,笔者根据自己的嵌入式学 ...
- RotateDrawable
用来对Drawable进行旋转,也是通过setLevel来控制旋转的,最大值也是:10000 相关属性如下: fromDegrees:起始的角度,,对应最低的level值,默认为0 toDegrees ...
- Oracle查询表和字段
查看表字段.类型.注释 SELECT A.COLUMN_NAME,B.comments,A.DATA_TYPE FROM USER_TAB_COLUMNS A LEFT JOIN user_col_c ...