Netty之ChannelHandler(三)
ChannelHandler是netty中的核心处理部分,我们使用netty的绝大部分代码都写在这部分,所以了解它的一些机制和特性是很有必要的。
一、Channel
Channel接口抽象了底层socket的一些状态属性以及调用方法

针对不同类型的socket提供不同的子类实现。

二、Channel生命周期

三、ChannelHandler
ChannelHandler用于处理Channel对应的事件
ChannelHandler接口里面只定义了三个生命周期方法,我们主要实现它的子接口ChannelInboundHandler和ChannelOutboundHandler,为了便利,框架提供了ChannelInboundHandlerAdapter,ChannelOutboundHandlerAdapter和ChannelDuplexHandler这三个适配类,在使用的时候只需要实现你关注的方法即可
ChannelHandler生命周期方法:

ChannelHandler里面定义三个生命周期方法,分别会在当前ChannelHander加入ChannelHandlerContext中,从ChannelHandlerContext中移除,以及ChannelHandler回调方法出现异常时被回调。
四、ChannelInboundHandler 和 ChannelOutboundHandler
1. ChannelInBoundHandler

介绍一下这些回调方法被触发的时机:
| 回调方法 | 触发时机 | client | server | 
|---|---|---|---|
| channelRegistered | 当前channel注册到EventLoop | true | true | 
| channelUnregistered | 当前channel从EventLoop取消注册 | true | true | 
| channelActive | 当前channel激活的时候 | true | true | 
| channelInactive | 当前channel不活跃的时候,也就是当前channel到了它生命周期尾 | true | true | 
| channelRead | 当前channel从远端读取到数据 | true | true | 
| channelReadComplete | channel read消费完读取的数据的时候被触发 | true | true | 
| userEventTriggered | 用户事件触发的时候 | ||
| channelWritabilityChanged | channel的写状态变化的时候触发 | 
可以注意到每个方法都带了ChannelHandlerContext作为参数,具体作用是,在每个回调事件里面,处理完成之后,使用ChannelHandlerContext的fireChannelXXX方法来传递给下个ChannelHandler,netty的codec模块和业务处理代码分离就用到了这个链路处理。
2. ChannelOutboundHandler

| 回调方法 | 触发时机 | client | server | 
|---|---|---|---|
| bind | bind操作执行前触发 | false | true | 
| connect | connect 操作执行前触发 | true | false | 
| disconnect | disconnect 操作执行前触发 | true | false | 
| close | close操作执行前触发 | false | true | 
| deregister | deregister操作执行前触发 | ||
| read | read操作执行前触发 | true | true | 
| write | write操作执行前触发 | true | true | 
| flush | flush操作执行前触发 | true | true | 
注意到一些回调方法有ChannelPromise这个参数,我们可以调用它的addListener注册监听,当回调方法所对应的操作完成后,会触发这个监听。
下面这个代码,会在写操作完成后触发,完成操作包括成功和失败:
public void write(ChannelHandlerContext ctx, Object msg, ChannelPromise promise) throws Exception {
    ctx.write(msg,promise);
    System.out.println("out write");
    promise.addListener(new GenericFutureListener<Future<? super Void>>() {
        @Override
        public void operationComplete(Future<? super Void> future) throws Exception {
            if(future.isSuccess()){
                System.out.println("OK");
            }
        }
    });
}
3. ChannelInBoundHandler 和 ChannelOutboundHandler的区别
个人感觉in和out的区别主要在于ChannelInboundHandler的channelRead和channelReadComplete回调和ChannelOutboundHandler的write和flush回调上,ChannelOutboundHandler的channelRead回调负责执行入栈数据的decode逻辑,ChannelOutboundHandler的write负责执行出站数据的encode工作。其他回调方法和具体触发逻辑有关,和in与out无关。
五、ChannelHandlerContext
每个ChannelHandler通过add方法加入到ChannelPipeline中去的时候,会创建一个对应的ChannelHandlerContext,并且绑定,ChannelPipeline实际维护的是ChannelHandlerContext 的关系。
在DefaultChannelPipeline源码中可以看到会保存第一个ChannelHandlerContext以及最后一个ChannelHandlerContext的引用。
六、总结
上述组件的关系:

- 每个Channel会绑定一个ChannelPipeline,ChannelPipeline中也会持有Channel的引用
 - ChannelPipeline持有ChannelHandlerContext链路,保留ChannelHandlerContext的头尾节点指针
 - 每个ChannelHandlerContext会对应一个ChannelHandler,也就相当于ChannelPipeline持有ChannelHandler链路
 - ChannelHandlerContext同时也会持有ChannelPipeline引用,也就相当于持有Channel引用
 - ChannelHandler链路会根据Handler的类型,分为InBound和OutBound两条链路
 
Netty之ChannelHandler(三)的更多相关文章
- Netty中的三种Reactor(反应堆)
		
目录: Reactor(反应堆)和Proactor(前摄器) <I/O模型之三:两种高性能 I/O 设计模式 Reactor 和 Proactor> <[转]第8章 前摄器(Proa ...
 - Netty 系列(三)Netty 入门
		
Netty 系列(三)Netty 入门 Netty 是一个提供异步事件驱动的网络应用框架,用以快速开发高性能.高可靠性的网络服务器和客户端程序.更多请参考:Netty Github 和 Netty中文 ...
 - Netty的ChannelHandler,ChannelHandlerContext,ChannelPipeline
		
本小节一起学习一下ChannelHandler,ChannelHandlerContext,ChannelPipeline这三个Netty常用的组件,不探究它们的底层源码,我们就简单的分析一下用法 首 ...
 - Netty入门(三)之web服务器
		
Netty入门(三)之web服务器 阅读前请参考 Netty入门(一)之webSocket聊天室 Netty入门(二)之PC聊天室 有了前两篇的使用基础,学习本文也很简单!只需要在前两文的基础上稍微改 ...
 - Netty Reator(三)Reactor 模型
		
Netty Reator(三)Reactor 模型 Netty 系列目录 (https://www.cnblogs.com/binarylei/p/10117436.html) 本文介绍 DC Sch ...
 - Netty 学习(三):通信协议和编解码
		
Netty 学习(三):通信协议和编解码 作者: Grey 原文地址: 博客园:Netty 学习(三):通信协议和编解码 CSDN:Netty 学习(三):通信协议和编解码 无论使用 Netty 还是 ...
 - 【Netty】ChannelHandler和ChannelPipeline
		
一.前言 前面学习了Netty的ByteBuf,接着学习ChannelHandler和ChannelPipeline. 二.ChannelHandler和ChannelPipeline 2.1 Cha ...
 - 【Netty】ChannelHandler和codec
		
一.前言 前面学习了Netty的codec框架,下面接着学习ChannelHandler与codec之间的关联. 二.ChannelHandler和codec Netty为不同的协议提供了处理器和编解 ...
 - Netty学习笔记(三)——netty源码剖析
		
1.Netty启动源码剖析 启动类: public class NettyNioServer { public static void main(String[] args) throws Excep ...
 
随机推荐
- Velocity与Jsp、Freemarker的对比
			
在java领域,表现层技术主要有三种:jsp.freemarker.velocity.jsp是大家最熟悉的技术 优点: 1.功能强大,可以写java代码 2.支持jsp标签(jsp tag) 3.支持 ...
 - react中使用动画
			
1. css原生动画的使用 import React, { useState } from "react" import "./index.css" funct ...
 - 洛谷P3702 [SDOI2017]序列计数
			
题目大意: Alice想要得到一个长度为\(n\)的序列,序列中的数都是不超过\(m\)的正整数,而且这\(n\)个数的和是\(p\)的倍数. Alice还希望,这\(n\)个数中,至少有一个数是质数 ...
 - powersploit使用实例
			
一.AntivirusBypass(绕过杀毒) Find-AVSignature 发现杀软的签名 1.先在靶机(windows 2008)上远程加载位于win7的Invoke-Shellcode.p ...
 - 【Java语言特性学习之三】Java4种对象引用
			
为了更灵活的控制对象的生命周期,在JDK1.2之后,引用被划分为(引用的级别和强度由高到低)强引用.软引用.弱引用.虚引用四种类型,每种类型有不同的生命周期,它们不同的地方就在于垃圾回收器对待它们会使 ...
 - java、JavaScript获取微信用户信息登录优化方案
			
1.获取微信用户信息要调用微信的好几个接口,再加上自己系统的接口就会变的很慢,影响用户体验,之前走过的弯路我就不赘述了,直接说新的方案. 2.第一步都是向微信发起获取用户code请求: 请求接口:ht ...
 - C#网页 截图
			
using System.IO; using System.Drawing; using System.Drawing.Imaging; using System.Threading; using S ...
 - js函数定义及一些说明
			
1.javascript定义函数的三种方法一.function语句//这个方法比较常用function fn(){ alert("这是使用function语句进行函数定义");}f ...
 - Java常用类Date相关知识
			
Date:类 Date 表示特定的瞬间,精确到毫秒. 在 JDK 1.1 之前,类 Date 有两个其他的函数.它允许把日期解释为年.月.日.小时.分钟和秒值.它也允许格式化和解析日期字符串. Dat ...
 - Lucene queryParser和analysis有什么不同?
			
在Lucene1.4.3中,queryParser和analysis分成独立的两个包,queryParser作用是解析查询语句,analysis作用是分词,queryParser在解析查询语句的时候会 ...