这篇随笔暂时不讲原理,首先搭建起一个简单的可以实现通信的Demo。之后的一系列随笔会进行一些原理上的分享。

不过在这之前大家最好了解一下Netty的线程模型和NIO编程模型,会对它的整体逻辑有所了解。

更新一篇关于NIO的博客:手动搭建I/O网络通信框架3:NIO编程模型,升级改造聊天室

首先创建好项目后在pom.xml引入Netty依赖

<dependency>
<groupId>io.netty</groupId>
<artifactId>netty-all</artifactId>
</dependency>

用Netty搭建一个WebSocket服务器整体上需要三样东西,不管是不是用的SpringBoot框架,这三样东西是必不可少的。

1.启动服务器的类(NettyServer),会进行一些初步的配置工作。

2.助手类(Handler),有自己定义的助手类,也有Netty提供的一些基本的助手类,比如对Http、WebSocket支持的助手类。

3.初始化器(Initializer),我们下面使用的是主从线程模型,从线程组里会分配出不同channel去处理不同客户端的请求,而每个channel里就会有各种助手类去实现一些功能。初始化器的作用就是对各种助手类进行绑定。

服务器启动类:

public class NettyServer {
private static int port; public NettyServer(int port) {
this.port = port;
}
public static void start() throws InterruptedException {//在main方法里调用这个方法,并用构造函数设置端口号
//创建主线程组,接收请求
EventLoopGroup bossGroup = new NioEventLoopGroup();
//创建从线程组,处理主线程组分配下来的io操作
EventLoopGroup workerGroup = new NioEventLoopGroup();
//创建netty服务器
try {
ServerBootstrap serverBootstrap = new ServerBootstrap();
serverBootstrap.group(bossGroup, workerGroup)//设置主从线程组
.channel(NioServerSocketChannel.class)//设置通道
.childHandler(new NettyServerInitializer());//子处理器,用于处理workerGroup中的操作
//启动server
ChannelFuture channelFuture = serverBootstrap.bind(port).sync();
//监听关闭channel
channelFuture.channel().closeFuture().sync();
} finally {
bossGroup.shutdownGracefully();//关闭主线程
workerGroup.shutdownGracefully();//关闭从线程
}
}
}

初始化器:

public class NettyServerInitializer extends ChannelInitializer<SocketChannel> {
@Override
protected void initChannel(SocketChannel socketChannel) throws Exception {
ChannelPipeline pipeline= socketChannel.pipeline();
//以下三个是Http的支持
//http解码器
pipeline.addLast(new HttpServerCodec());
//支持写大数据流
pipeline.addLast(new ChunkedWriteHandler());
//http聚合器
pipeline.addLast(new HttpObjectAggregator(1024*62));
//websocket支持,设置路由
pipeline.addLast(new WebSocketServerProtocolHandler("/ws"));
//添加自定义的助手类
pipeline.addLast(new NettyHandler());
}
}

自定义助手类:

这个类就是业务的核心,客户端的请求会在这里处理。比如客户端连接、客户端发送消息、给客户端发送消息等等。

自定义助手类需要重写的方法可以根据自己的需求重写,这里就不把每个方法都重写一遍了,完整的大家可以去找找文档看看。

如果需要在助手类中用到@Autowire注解,可以参考这个博客,网上有很多说明,这里就不再重复了https://blog.csdn.net/weixin_30828379/article/details/95009595

public class NettyHandler extends SimpleChannelInboundHandler<TextWebSocketFrame> {//TextWebSocketFrame是netty用于处理websocket发来的文本对象
  //所有正在连接的channel都会存在这里面,所以也可以间接代表在线的客户端
public static ChannelGroup channelGroup = new DefaultChannelGroup(GlobalEventExecutor.INSTANCE);
  //在线人数
public static int online;
//接收到客户都发送的消息
@Override
public void channelRead0(ChannelHandlerContext ctx, TextWebSocketFrame msg) throws Exception {
SendAllMessages(ctx,send_message);//send_message是我的自定义类型,前后端分离往往需要统一数据格式,可以先把对象转成json字符串再发送给客户端
}
//客户端建立连接
@Override
public void handlerAdded(ChannelHandlerContext ctx) throws Exception {
channelGroup.add(ctx.channel());
online=channelGroup.size();
System.out.println(ctx.channel().remoteAddress()+"上线了!");
}
//关闭连接
@Override
public void handlerRemoved(ChannelHandlerContext ctx) throws Exception {
channelGroup.remove(ctx.channel());
online=channelGroup.size();
System.out.println(ctx.channel().remoteAddress()+"断开连接");
} //出现异常
@Override
public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {
cause.printStackTrace();
} //给某个人发送消息
private void SendMessage(ChannelHandlerContext ctx, Send_Message msg) {
ctx.channel().writeAndFlush(new TextWebSocketFrame(JSON.toJSONString(msg)));
} //给每个人发送消息,除发消息人外
private void SendAllMessages(ChannelHandlerContext ctx,Send_Message msg) {
for(Channel channel:channelGroup){
if(!channel.id().asLongText().equals(ctx.channel().id().asLongText())){
channel.writeAndFlush(new TextWebSocketFrame(JSON.toJSONString(msg)));
}
}
}
}

前端创建WebSocket对象进行访问

通过socket就可以用一些api进行发送消息,接收消息的操作。然后把接收的数据按各自的需求展示出来就行了,前端部分就不再赘述了。

下面8088端口记得要在main方法里面设置。

if (window.WebSocket) {
var host = window.location.hostname;
var url = "ws://" + host + ":8088/ws";
var socket = new WebSocket(url);
}else{
alert("你的浏览器不支持WebSocket。请不要使用低版本的IE浏览器。");
}

最新文章

  1. 我为Net狂 ~ 社交平台系列小集合!
  2. [转载]Docker的安装配置及使用详解
  3. Asp.Net MVC&lt;二&gt; : IIS/asp.net管道
  4. asp获取虚拟目录根路径
  5. [C++] 如何查看DLL有哪些函数
  6. UI Button
  7. MYSQL中 ENUM、SET 类型(建议用tinyint代替)
  8. ASCII码常用值
  9. 抛出异常的区别 throw 和throw ex
  10. java中的public,protected,private权限修饰
  11. visual studio 中将选中代码相同的代码的颜色设置,修改高亮颜色
  12. Mac下Git安装及配置
  13. 03安卓TextView
  14. js中slice,SubString和SubStr的区别
  15. 【zabbix教程系列】一、初识zabbix
  16. Ubuntu 清除缓存 apt-get命令参数
  17. 我为什么要谈KeepAlive(文末增加nginx 负载tcp长连接保持 demo)
  18. MySQL字符集 utf8 和 utf8mb4 区别及排序规则 general_ci 和 unicode_ci 和 bin 的区别
  19. R语言-图的要素颜色
  20. 10. Halloween 万圣节

热门文章

  1. 使用Python中的NLTK和spaCy删除停用词与文本标准化
  2. UVA11987 Almost Union-Find 并查集的节点删除
  3. POJ 1797 最短路变形所有路径最小边的最大值
  4. coding++:MySQL-ERROR:Column &#39;complaint_settlement_id&#39; in field list is ambiguous
  5. Input标签中属性的注意点
  6. [vijos]1051送给圣诞夜的极光&lt;BFS&gt;
  7. CoderForces 327D Block Tower
  8. 一位读者刚刚收割阿里、腾讯等大厂Offer,他说这些话一定要和你们说一下
  9. STM32CubeMX的使用
  10. 第一讲:Git分区,配置与日志