Netty源码分析第7章(编码器和写数据)----第2节: MessageToByteEncoder
Netty源码分析第七章: Netty源码分析
第二节: MessageToByteEncoder
同解码器一样, 编码器中也有一个抽象类叫MessageToByteEncoder, 其中定义了编码器的骨架方法, 具体编码逻辑交给子类实现
解码器同样也是个handler, 将写出的数据进行截取处理, 我们在学习pipeline中我们知道, 写数据的时候会传递write事件, 传递过程中会调用handler的write方法, 所以编码器码器可以重写write方法, 将数据编码成二进制字节流然后再继续传递write事件
首先看MessageToByteEncoder的类声明:
public abstract class MessageToByteEncoder<I> extends ChannelOutboundHandlerAdapter{//省略类体 }
这里继承ChannelOutboundHandlerAdapter, 说明是个outBoundhandler, 我们知道write事件是个outBound事件, 而outBound事件只能通过outBoundHandler进行传输
write事件传播过程中要调用handler的write方法
我们跟到MessageToByteEncoder的write方法中:
public void write(ChannelHandlerContext ctx, Object msg, ChannelPromise promise) throws Exception {ByteBuf buf = null;try {if (acceptOutboundMessage(msg)) {@SuppressWarnings("unchecked")I cast = (I) msg;buf = allocateBuffer(ctx, cast, preferDirect);try {encode(ctx, cast, buf);} finally {ReferenceCountUtil.release(cast);}if (buf.isReadable()) {ctx.write(buf, promise);} else {buf.release();ctx.write(Unpooled.EMPTY_BUFFER, promise);}buf = null;} else {ctx.write(msg, promise);}} catch (EncoderException e) {throw e;} catch (Throwable e) {throw new EncoderException(e);} finally {if (buf != null) {buf.release();}} }
首先通过 if (acceptOutboundMessage(msg)) 判断当前对象是否可处理
如果可处理, 则进入if块中的逻辑, 如果不能处理, 则进入else块, 通过ctx.write(msg, promise)继续传递write事件
我们看if块中
I cast = (I) msg 这里是强制类型转换, 转换成I类型, I类型是个泛型, 具体类型由用户定义
buf = allocateBuffer(ctx, cast, preferDirect) 这里进行缓冲区分配
跟到allocateBuffer方法中:
protected ByteBuf allocateBuffer(ChannelHandlerContext ctx, @SuppressWarnings("unused") I msg, boolean preferDirect) throws Exception {if (preferDirect) {return ctx.alloc().ioBuffer();} else {return ctx.alloc().heapBuffer();} }
这里会直接通过ctx的内存分配器进行内存分配, 通过判断preferDirect来分配堆内存或者堆外内存, 默认情况下是分配堆外内存
有关内存分配, 我们之前已经做过相关的剖析
回到write方法中:
内存分配结束之后会调用encode(ctx, cast, buf)方法进行编码, 该类由子类实现
子类可以通过继承该类, 重写encode方法, 将参数对象cast编码成字节写入到传入的ByteBuf中, 就完成了编码工作
编码完成后后, 会通过ReferenceCountUtil.release(cast)将cast对象释放
if (buf.isReadable()) 这里判断buf是否有可读字节, 如果有可读字节, 则继续传递write事件
如果没有可读字节, 则将buf进行释放, 继续传播write事件, 传递一个空的ByteBuf
最后将buf设置为空
以上就是有关抽象编码器的抽象逻辑, 具体的编码逻辑还需要其子类去做
上一节: writeAndlush事件传播
下一节: 写buffer队列
转载于:https://www.cnblogs.com/xiangnan6122/p/10208131.html
Netty源码分析第7章(编码器和写数据)----第2节: MessageToByteEncoder相关推荐
- Netty源码分析第6章(解码器)----第4节: 分隔符解码器
Netty源码分析第6章(解码器)---->第4节: 分隔符解码器 Netty源码分析第六章: 解码器 第四节: 分隔符解码器 基于分隔符解码器DelimiterBasedFrameDecode ...
- Netty源码分析第1章(Netty启动流程)----第4节: 注册多路复用
Netty源码分析第1章(Netty启动流程)---->第4节: 注册多路复用 Netty源码分析第一章:Netty启动流程 第四节:注册多路复用 回顾下以上的小节, 我们知道了channe ...
- Netty源码分析第5章(ByteBuf)----第5节: directArena分配缓冲区概述
Netty源码分析第5章(ByteBuf)---->第5节: directArena分配缓冲区概述 Netty源码分析第五章: ByteBuf 第五节: directArena分配缓冲区概述 上 ...
- Flume 1.7 源码分析(四)从Source写数据到Channel
Flume 1.7 源码分析(一)源码编译 Flume 1.7 源码分析(二)整体架构 Flume 1.7 源码分析(三)程序入口 Flume 1.7 源码分析(四)从Source写数据到Channe ...
- Flume 1.7 源码分析(五)从Channel获取数据写入Sink
Flume 1.7 源码分析(一)源码编译 Flume 1.7 源码分析(二)整体架构 Flume 1.7 源码分析(三)程序入口 Flume 1.7 源码分析(四)从Source写数据到Channe ...
- Netty源码分析系列之常用解码器(下)——LengthFieldBasedFrameDecoder
扫描下方二维码或者微信搜索公众号菜鸟飞呀飞,即可关注微信公众号,Spring源码分析和Java并发编程文章. 前言 在上一篇文章中分析了三个比较简单的解码器,今天接着分析最后一个常用的解码器:Leng ...
- 【Netty源码分析摘录】(八)新连接的接入
文章目录 1.问题 2.检测新连接接入 3.创建客户端 channel 4. 绑定 NioEventLoop 4.1 register0 4.1.1 doRegister() 4.1.2 pipeli ...
- Netty源码分析(六)—Future和Promis分析
Netty源码分析(六)-Future和Promis分析 Future用来在异步执行中获取提前执行的结果 个人主页:tuzhenyu's page 原文地址:Netty源码分析(六)-Future和P ...
- Netty源码分析系列之服务端Channel的端口绑定
扫描下方二维码或者微信搜索公众号菜鸟飞呀飞,即可关注微信公众号,Spring源码分析和Java并发编程文章. 微信公众号 问题 本文内容是接着前两篇文章写的,有兴趣的朋友可以先去阅读下两篇文章: Ne ...
最新文章
- linux存储--从内核文件系统看文件读写过程(四)
- Redis复制的高可用详解
- linux的tmp文件夹定期会删除么,关于linux tmp下文件自动删除的问题
- ntv.js框架(第三章) - 机顶盒HTML和CSS编写注意事项
- EL表达式的作用与限制条件
- Mysql一主多从和读写分离配置简记
- IDEA 配置Maven国内源
- 微信公众平台开发(112) 微信卡券
- C++ 对象的内存布局(下)
- 业务模式制胜,BLM战略规划七步法
- php 数据库查重,使用SimHash进行海量内容数据查重 - PHP版
- Mocha Pro 2022 Plugins for mac(达芬奇/Fusion/Nuke/OFX插件)
- 联想启天m410进bios_联想启天M410台式机重装系统win7-联想win7系统下载
- ISP许可证是什么证?
- python excel行列转置_Excel 行列转换的最简方法
- linux0.11主存管理程序阅读注释笔记
- ubuntu查看 固态硬盘位置_在Ubuntu(Linux)中启用固态硬盘(固态硬盘)TRIM | MOS86...
- Axure与markman
- raid5+1和raid5有什么区别?raid5最多可以上几块硬盘
- 爆乱媒评——雄性传媒,大与小的技巧——凤凰卫视VS央视评点(1)
热门文章
- JDK1.7安装配置环境变量+图文说明Jmeter安装
- 线程同步之——互斥量及死锁问题
- CSS Specificity--CSS特性、权重、优先级---CSS specificity规则、
- 利用优盘安装win2008r2系统
- 第一章 OSI参考模型
- 【数据平台】Python解析Ngnix日志
- 机器学习知识点(三十一)LDA数学八卦
- PHP中htmlentities和htmlspecialchars的区别
- webpack 中的加载器简介||webpack 中加载器的基本使用——1. 打包处理 css 文件 2. 打包处理 less 文件 3.打包处理 scss 文件
- 转载:python3 安装pycrypto