netty自定义channel id、netty custom channel id

搞搞netty时发现默认的id很长,无法直接自定义。
于是我网上搜索了search一下,发现没有相关文章,那就自己看看源码手撸一个实现。这难不倒拥有7年代码经验的我,通过本文章你能大概学到如何根据源码定制功能。

通过netty官网说明唯一id:https://netty.io/wiki/new-and-noteworthy-in-4.1.html

全局唯一通道ID
每个频道现在都有一个全局唯一的ID,该ID由以下内容生成: 优选全局唯一的MAC地址(EUI-48或EUI-64),
当前进程ID,
系统#currentTimeMillis()
系统#nanoTime()
随机32位整数
顺序递增的32位整数
可以使用信道获得信道的ID.id()方法。

默认的id是这样的:通道id:a85e45fffec07f9b-00002454-00000000-084db96c90765225-d150e0b4
虽然很唯一,但不符合我们的系统,应该将它自定义。

一、自定义ID试调源码过程

我是基于netty 4.1.79.Final 2022年8月12日最新版本。
首先看id的实现接口:IDEA中按Ctrl + Alt + O 搜索 channel ID

应该就是这个了ChannelId接口了,接着看他的实现类


应该就是默认的实现了
观察到id为空时他就会new一个

接着看一下哪里用他生成:
此时我们发现是一个构造类用到了他AbstractChannelAbstractChannel应该就是所有管道类的父类。

他的构造方法中也newId()

他就是在这里吧ID给生成出来:

框架开发一般规则/套路:不直接使用构造类,那么我们看看他的继承使用情况:

太多了,我们不知道哪个会加载。
回到我们的客户端初始化类,看看能不能在初始化配置时找到自定义他的地方:

此时发现打印的是class io.netty.channel.socket.nio.NioSocketChannel这个类,恰好对应上面的NioSocketChannel处理
NioSocketChannel这个是客户端的管道处理类,默认是使用socket协议。恰好发现它是继承了AbstractChannel

二、走查源码自定义结果

那应该就是通过自定义NioSocketChannel这个类进行自定义id生成。

三、实现自定义

通过上面的结果,我们通过继承NioSocketChannel来实现自定义id生成:使用UUID

import io.netty.channel.ChannelId;
import io.netty.channel.socket.nio.NioSocketChannel; import java.util.UUID; /**
* @author lingkang
* Created by 2022/8/12
*/
public class MyNioSocketChannel extends NioSocketChannel { protected ChannelId newId() {
ChannelId channelId = new ChannelId() {
@Override
public String asShortText() {
return UUID.randomUUID().toString();
} @Override
public String asLongText() {
return UUID.randomUUID().toString();
} @Override
public int compareTo(ChannelId o) {
return 0;
}
};
return channelId;
}
}

客户端初始化那就是用我们集成自定义的类

运行结果:

这个自定义是正确的,所以服务端也能按照上面的思路进行自定义。

四、完整代码:

import io.netty.channel.ChannelId;
import io.netty.channel.socket.nio.NioSocketChannel; import java.util.UUID; /**
* @author lingkang
* Created by 2022/8/12
*/
public class MyNioSocketChannel extends NioSocketChannel { protected ChannelId newId() {
ChannelId channelId = new ChannelId() {
@Override
public String asShortText() {
return UUID.randomUUID().toString();
} @Override
public String asLongText() {
return UUID.randomUUID().toString();
} @Override
public int compareTo(ChannelId o) {
return 0;
}
};
return channelId;
}
}
import io.netty.bootstrap.ServerBootstrap;
import io.netty.buffer.Unpooled;
import io.netty.channel.*;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.SocketChannel;
import io.netty.channel.socket.nio.NioServerSocketChannel;
import top.lingkang.flychat.common.MsgBody;
import top.lingkang.flychat.common.code.RpcDecoder;
import top.lingkang.flychat.common.code.RpcEncoder;
import top.lingkang.flychat.server.ServerHandler;
import top.lingkang.flychat.server.ServerInit; import java.util.Date; /**
* @author lingkang
* Created by 2022/8/12
*/
public class Test01Server {
public static void main(String[] args) {
EventLoopGroup bossGroup = new NioEventLoopGroup(1);
EventLoopGroup workGroup = new NioEventLoopGroup();
try {
ServerBootstrap serverBootstrap = new ServerBootstrap();
serverBootstrap.group(bossGroup, workGroup)
.channel(NioServerSocketChannel.class)
// 当服务器请求处理线程全满时,用于临时存放已完成三次握手的请求的队列的最大长度。
.option(ChannelOption.SO_BACKLOG, 50)
// .childOption(ChannelOption.SO_KEEPALIVE, true)
.option(ChannelOption.CONNECT_TIMEOUT_MILLIS, 5000)
.childHandler(new ChannelInitializer<SocketChannel>(){
@Override
protected void initChannel(SocketChannel ch) throws Exception {
ch.pipeline()
.addFirst("encode", new RpcEncoder(MsgBody.class))//编码器
.addFirst("decode", new RpcDecoder(MsgBody.class))//解码器
.addLast(new ChannelInboundHandlerAdapter(){
public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception {
MsgBody body = (MsgBody) msg;
System.out.println("接收到Client端信息:" + body.toString());
//返回的数据结构
MsgBody response = new MsgBody();
response.setCode(200);
response.setData(new Date());
response.setMsg("server响应结果");
System.out.println("server thread id=" + Thread.currentThread().getId());
ctx.writeAndFlush(response);
} @Override
public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {
cause.printStackTrace();
closeOnFlush(ctx.channel());
} private void closeOnFlush(Channel ch) {
if (ch.isActive()) {
ch.writeAndFlush(Unpooled.EMPTY_BUFFER).addListener(ChannelFutureListener.CLOSE);
}
} @Override
public void handlerRemoved(ChannelHandlerContext ctx) throws Exception {
super.handlerRemoved(ctx);
System.out.println("有链接断开");
}
});
}
}); //启动同步监听
serverBootstrap.bind("127.0.0.1", 8081).sync().channel().closeFuture().sync();
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
workGroup.shutdownGracefully();
bossGroup.shutdownGracefully();
}
}
}
import io.netty.bootstrap.Bootstrap;
import io.netty.buffer.Unpooled;
import io.netty.channel.*;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.SocketChannel;
import io.netty.channel.socket.nio.NioSocketChannel;
import top.lingkang.flychat.common.MsgBody;
import top.lingkang.flychat.common.code.RpcDecoder;
import top.lingkang.flychat.common.code.RpcEncoder; import java.util.Date; /**
* @author lingkang
* Created by 2022/8/12
*/
public class Test01Client {
public static void main(String[] args) throws Exception {
Bootstrap bootstrap = new Bootstrap();
EventLoopGroup group = new NioEventLoopGroup(1);
try {
bootstrap
.group(group)
.channel(MyNioSocketChannel.class)// 使用NioSocketChannel来作为连接用的channel类
.handler(new ChannelInitializer<SocketChannel>() {
@Override
protected void initChannel(SocketChannel ch) throws Exception {
System.out.println(ch.getClass());
ch.pipeline()
.addFirst("encode", new RpcEncoder(MsgBody.class))//编码器
.addFirst("decode", new RpcDecoder(MsgBody.class))//解码器
.addLast(new ChannelInboundHandlerAdapter() {
@Override
public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception {
MsgBody body = (MsgBody) msg; System.out.println("接收到Server端响应消息:" + body.toString());
// throw new RuntimeException("手动抛出");
} @Override
public void channelInactive(ChannelHandlerContext ctx) throws Exception {
closeOnFlush(ctx.channel());
} @Override
public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {
cause.printStackTrace();
closeOnFlush(ctx.channel());
} private void closeOnFlush(Channel ch) {
if (ch.isActive()) {
ch.writeAndFlush(Unpooled.EMPTY_BUFFER).addListener(ChannelFutureListener.CLOSE);
}
}
});
}
}); // Start the connection attempt.
Channel channel = bootstrap.connect("127.0.0.1", 8081).sync().channel();
System.out.println("连接服务器成功");
int i = 0;
while (true) {
try {
//每2秒给服务器发一次数据
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
}
i++;
channel.writeAndFlush(new MsgBody(200, "客户端给服务端发消息:" + i, new Date()));
if (i == 5) {
System.out.println("通道id:" + channel.id().asLongText());
channel.close();
break;
}
}
} finally {
//关闭线程组
group.shutdownGracefully();
}
}
}

netty自定义channel id的更多相关文章

  1. netty 自定义协议

    netty 自定义协议 netty 是什么呢? 相信很多人都被人问过这个问题.如果快速准确的回复这个问题呢?网络编程框架,netty可以让你快速和简单的开发出一个高性能的网络应用.netty是一个网络 ...

  2. 项目系统Netty的Channel和用户之间的关系绑定正确做法,以及Channel通道的安全性方案

    前言 考虑一个功能业务,在web程序中向指定的某个用户进行实时通讯 在Web运用的Socket通讯功能中(如在线客服),为保证点对点通讯.而这个看似简单的根据用户寻到起channel通道实际会碰到不少 ...

  3. netty自定义解码器

    在socket传输通信中容易丢包问题,什么半包问题,这些都是很正常的问题,处理方法就是定义自己的编解码规则了,让每次接收按定义好的规则为一个完整包作为数据源即可. 下面个例子就是netty自定义的一个 ...

  4. Netty之Channel*

    Netty之Channel* 本文内容主要参考**<<Netty In Action>> ** 和Netty的文档和源码,偏笔记向. 先简略了解一下ChannelPipelin ...

  5. Netty 源码解析(二):Netty 的 Channel

    本文首发于微信公众号[猿灯塔],转载引用请说明出处 接下来的时间灯塔君持续更新Netty系列一共九篇 Netty源码解析(一):开始 当前:Netty 源码解析(二): Netty 的 Channel ...

  6. Netty自定义协议解析原理与应用

    目前,大家都选择Netty做为游戏服务器框架网络通信的框架,而且目前也有很多优秀的产品是基于Netty开发的.它的稳定性,易用性和高效率性已得到广泛的认同.在游戏服务器开发中,选择netty一般就意味 ...

  7. spark2.1源码分析3:spark-rpc如何实现将netty的Channel隐藏在inbox中

    class TransportServer bootstrap.childHandler(new ChannelInitializer<SocketChannel>() { @Overri ...

  8. spark-rpc是如何实现将netty的Channel隐藏在inbox中的

    class TransportServer bootstrap.childHandler(new ChannelInitializer<SocketChannel>() { @Overri ...

  9. Netty:Channel 建立后消息发送失败

    1. 问题现象 Channel 建立后消息发送失败: ChannelFuture future = DeviceManager.getBootstrap().connect(); deviceChan ...

  10. Netty自定义Encoder/Decoder进行对象传递

    转载:http://blog.csdn.net/top_code/article/details/50901623 在上一篇文章中,我们使用Netty4本身自带的ObjectDecoder,Objec ...

随机推荐

  1. 一文带你实现云上部署轻量化定制表单Docker

    本文分享自华为云社区 <[华为云云耀云服务器L实例评测|云原生]自定制轻量化表单Docker快速部署云耀云服务器 | 玩转华为云>,作者:计算机魔术师. 华为云的云耀云服务器L实例备受推崇 ...

  2. 为什么 CSS flex 布局中没有 `justify-items` 和 `justify-self`?

    为什么 CSS flex 布局中没有 justify-items 和 justify-self? 为什么在 CSS flex 布局中存在 align-items 和 align-self,却没有 ju ...

  3. 轻巧的批量图片压缩工具imgfast

    现在的手机拍照动辄2M3M,还有7M8m的,如果要把这些文件上传到网上应用,浪费网络,占用资源 所以2022年中秋写了这个小工具,可以批量进行图片文件压缩,支持jpg和png. 文件下载链接https ...

  4. React框架的基本运行原理与组件定义方式

    React框架的基本运行原理 React的本质是内部维护了一套虚拟DOM树,这个虚拟DOM树就是一棵js对象树,它和真实DOM树是一致的,一一对应的. 当某一个组件的state发生修改时,就会生成一个 ...

  5. 【虹科干货】Redis Enterprise vs ElastiCache——如何选择缓存解决方案?

    使用Redis 或 Amazon ElastiCache 来作为缓存加速已经是业界主流的解决方案,二者各有什么优势?又有哪些区别呢? 为了提高 Web 应用程序和数据驱动服务的性能与效率,使用 Red ...

  6. idea 连接远程 docker 并部署项目到 docker

    目录 idea 连接远程 docker 1. 安装 docker 插件 2. 登录远程服务器,修改docker配置 3. 添加云服务器防火墙规则 4. idea 配置连接 docker 部署项目到 d ...

  7. QT(2)-QRegExp

    QT(2)-QRegExp 1 正则表达式 正则表达式--详情版+常用表达式 Qt中正则表达式(常用) Qt 正则表达式介绍 QRegExp的使用 2 QRegExp 2.1 indexIn int ...

  8. vue 中引入pingfang字体 或者其他字体 支持ttf otf格式

    新建一个font 文件 里面放字体文件 可以百度搜索你想要的字体下载下来 一般10m左右 新建一个font.css 里面配置字体 @font-face {     font-family: 'PF'; ...

  9. typeid关键词

    typeid是c++的关键字,typeid操作符的返回结果是名为type_info的标准库类型的对象的引用(在头文件typeinfo中定义) ISO C++标准并没有确切定义type_info,它的确 ...

  10. 递归+记忆化递归+DP:斐波那契数列

    递归:算法复杂度O(2^N) 1 int fib(int n) 2 { 3 if (n == 0) 4 { 5 return 0; 6 } 7 if (n == 1) 8 { 9 return 1; ...