当前位置:   article > 正文

使用 Netty 实现基于 CS 架构的 WebSocket 功能_cs架构全双工通信

cs架构全双工通信

一 点睛

WebSocket 是 HTML5 的一种协议,它可以实现浏览器与服务器全双工通信。Websocket 是应用层第七层上的一个应用层协议,它必须依赖 HTTP 协议进行一次握手,握手成功后,数据就可以直接使用 TCP 通道传输,之后就与 HTTP 无关。简单地讲,开始握手需要借助 HTTP 请求完成,之后就会升级为 WebSocket 协议。

本篇开发一个基于 WebSocket 协议的长连接通信。 

采用 Netty 作为长连接的服务端,使用浏览器作为客户端,二者交互图如下。

二 服务端

1 主程序类

  1. package netty.websocket;
  2. import io.netty.bootstrap.ServerBootstrap;
  3. import io.netty.channel.ChannelFuture;
  4. import io.netty.channel.EventLoopGroup;
  5. import io.netty.channel.nio.NioEventLoopGroup;
  6. import io.netty.channel.socket.nio.NioServerSocketChannel;
  7. public class MyNettyServerTest {
  8. public static void main(String[] args) {
  9. EventLoopGroup bossGroup = new NioEventLoopGroup();
  10. EventLoopGroup workerGroup = new NioEventLoopGroup();
  11. try {
  12. // ServerBootstrap:服务端启动时的初始化操作
  13. ServerBootstrap serverBootstrap = new ServerBootstrap();
  14. // 将 bossGroup 和 workerGroup 注册到服务端的 Channel上,并注册一个服务端的初始化器 NettyServerInitializer
  15. // 该初始化器中的 initChannel() 方法,会在连接被注册后立刻执行;最后将端口号绑定到 8888
  16. ChannelFuture channelFuture = serverBootstrap
  17. .group(bossGroup, workerGroup)
  18. .channel(NioServerSocketChannel.class)
  19. .childHandler(new MyNettyServerInitializer())
  20. .bind(8888).sync();
  21. channelFuture.channel().closeFuture().sync();
  22. } catch (Exception e) {
  23. e.printStackTrace();
  24. } finally {
  25. bossGroup.shutdownGracefully();
  26. workerGroup.shutdownGracefully();
  27. }
  28. }
  29. }

2 自定义初始化器

  1. package netty.websocket;
  2. import io.netty.channel.ChannelInitializer;
  3. import io.netty.channel.ChannelPipeline;
  4. import io.netty.channel.socket.SocketChannel;
  5. import io.netty.handler.codec.http.HttpObjectAggregator;
  6. import io.netty.handler.codec.http.HttpServerCodec;
  7. import io.netty.handler.codec.http.websocketx.WebSocketServerProtocolHandler;
  8. public class MyNettyServerInitializer extends ChannelInitializer<SocketChannel> {
  9. protected void initChannel(SocketChannel sc) throws Exception {
  10. ChannelPipeline pipeline = sc.pipeline();
  11. pipeline.addLast("HttpServerCodec", new HttpServerCodec());
  12. /*
  13. HttpObjectAggregator:把多个 HttpMessage 组装成一个完整的 Http 请求(FullHttpRequest)或者响应(FullHttpResponse)。
  14. */
  15. pipeline.addLast("HttpObjectAggregator", new HttpObjectAggregator(4096));
  16. // 处理 websocket 的 netty 处理器,可以通过构造方法绑定 webSocket 的服务端地址
  17. pipeline.addLast("WebSocketServerProtocolHandler", new WebSocketServerProtocolHandler("/myWebSocket"));
  18. // 自定义处理器
  19. pipeline.addLast("MyNettyServerHandler", new MyNettyServerHandler());
  20. }
  21. }

3 自定义处理器

  1. package netty.websocket;
  2. import io.netty.channel.ChannelHandlerContext;
  3. import io.netty.channel.SimpleChannelInboundHandler;
  4. import io.netty.handler.codec.http.websocketx.TextWebSocketFrame;
  5. // 泛型 TextWebSocketFrame:WebSocket 处理的处理文本类型
  6. public class MyNettyServerHandler extends SimpleChannelInboundHandler<TextWebSocketFrame> {
  7. // 接收 WebSocket 客户端发送来的数据(Websocket以 fram e的形式传递数据)
  8. @Override
  9. protected void channelRead0(ChannelHandlerContext ctx, TextWebSocketFrame frame) {
  10. System.out.println("Server收到消息:" + frame.text());
  11. // 向 WebSocket 客户端发送数据
  12. ctx.channel().writeAndFlush(new TextWebSocketFrame("helo client..."));
  13. }
  14. @Override
  15. public void handlerAdded(ChannelHandlerContext ctx) throws Exception {
  16. System.out.println("客户端加入:id=" + ctx.channel().id());
  17. }
  18. @Override
  19. public void handlerRemoved(ChannelHandlerContext ctx) throws Exception {
  20. System.out.println("客户端离开:id=" + ctx.channel().id());
  21. }
  22. }

三 客户端

1 html 页面

  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4. <meta charset="UTF-8">
  5. <title>Title</title>
  6. <script type="text/javascript">
  7. var webSocket = new WebSocket("ws://localhost:8888/myWebSocket");
  8. //检测WebSocket服务端是否开启
  9. webSocket.onopen = function (event) {
  10. document.getElementById("tip").innerText = "连接开启";
  11. }
  12. //检测WebSocket服务端是否关闭
  13. webSocket.onclose = function (event) {
  14. document.getElementById("tip").innerText = "连接关闭";
  15. }
  16. //接收WebSocket服务端发送来的数据(数据保存在event对象中)
  17. webSocket.onmessage = function (event) {
  18. document.getElementById("tip").innerText = "接收到的服务端消息:" + event.data;
  19. }
  20. function sendMessage(msg) {
  21. if (webSocket.readyState == WebSocket.OPEN) {
  22. //向WebSocket服务端发送数据
  23. webSocket.send(msg);
  24. }
  25. }
  26. </script>
  27. </head>
  28. <body onload="init()">
  29. <form>
  30. <textarea name="message"></textarea> <br/>
  31. <input type="button" onclick="sendMessage(this.form.message.value)" value="向服务端发送WebSocket数据"/>
  32. </form>
  33. <div id="tip"></div>
  34. </body>
  35. </html>

四 测试

1 服务端

客户端加入:id=0ecbc18d

Server收到消息:hello,我是客户端

客户端离开:id=0ecbc18d

2 客户端

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/2023面试高手/article/detail/107337
推荐阅读
相关标签
  

闽ICP备14008679号