当前位置: 首页 > news >正文

Netty + WebSocket:搭建快速且稳定的双向通信通道

在这里插入图片描述

前言

在现代应用中,实时通信已经成为了非常重要的需求,无论是在即时聊天、在线游戏,还是金融、物联网等领域,双向实时通信都是保证用户体验的关键。传统的HTTP请求-响应方式已经不再适应这种需求,因此,我们需要一种更高效、实时的通信方式,而 WebSocket 就是为此应运而生的协议。

在这篇博客中,我们将介绍如何基于 NettyWebSocket 搭建一个快速且稳定的双向通信通道。

一、WebSocket 简介

WebSocket 是 HTML5 引入的协议,旨在建立持久的全双工通信。与传统的 HTTP 不同,WebSocket 协议可以在客户端和服务器之间建立一个持久的连接,并且客户端和服务器可以通过该连接随时双向通信。

WebSocket 的优势:

  • 双向通信:客户端和服务器都可以主动发送消息。

  • 低延迟:一旦建立连接,消息可以实时传递,避免了 HTTP 请求的延迟。

  • 节省带宽:WebSocket 不需要像 HTTP 一样每次请求都携带头信息,减少了数据的冗余。

二、为什么选择 Netty?

Netty 是一个基于 Java 的高性能网络通信框架。其特点包括:

高性能:Netty 基于 NIO(非阻塞 IO)进行设计,具有很高的并发处理能力。

模块化设计:Netty 提供了许多功能模块,能够轻松扩展和定制。

广泛应用:它被广泛应用于各种高并发、高性能的网络通信场景,如即时消息系统、视频流、金融交易系统等。

利用 Netty 搭建 WebSocket 服务,不仅能够提供高性能的网络处理能力,还能够灵活地处理连接、消息解析、线程管理等问题,非常适合需要高吞吐量和低延迟的应用场景。

三、搭建基于 Netty 和 WebSocket 的双向通信

1. 添加依赖

首先,我们需要添加 Netty 和 WebSocket 相关的依赖。可以通过 Maven 或 Gradle 来引入。

 <properties><maven.compiler.source>17</maven.compiler.source><maven.compiler.target>17</maven.compiler.target><project.build.sourceEncoding>UTF-8</project.build.sourceEncoding><netty.version>4.2.4.Final</netty.version></properties><dependencies><dependency><groupId>io.netty</groupId><artifactId>netty-all</artifactId><version>${netty.version}</version></dependency><dependency><groupId>org.java-websocket</groupId><artifactId>Java-WebSocket</artifactId><version>1.6.0</version></dependency></dependencies>

2. WebSocket 服务器端实现

接下来,我们开始编写 WebSocket 服务器端代码。在 Netty 中,所有的网络通信都是基于 Channel 的,这使得它非常适合用于 WebSocket 的实现。

WebSocket 服务器端的实现步骤

  • 创建一个 WebSocketServerHandler:负责处理客户端连接的建立、消息的接收和发送。
  • 配置 Netty 的 EventLoopGroup 和 Channel:用于启动服务器,处理网络事件。
  • 启动 WebSocket 服务:通过 Netty 启动 WebSocket 服务器。
package org.example.websocket.server;import io.netty.bootstrap.ServerBootstrap;
import io.netty.buffer.Unpooled;
import io.netty.channel.*;
import io.netty.channel.nio.NioIoHandler;
import io.netty.channel.socket.nio.NioServerSocketChannel;
import io.netty.handler.codec.http.FullHttpRequest;
import io.netty.handler.codec.http.HttpMethod;
import io.netty.handler.codec.http.HttpObjectAggregator;
import io.netty.handler.codec.http.HttpServerCodec;
import io.netty.handler.codec.http.websocketx.*;
import io.netty.handler.stream.ChunkedWriteHandler;
import io.netty.util.CharsetUtil;public class WebSocketServer {public static void main(String[] args) throws InterruptedException {// 创建两个 EventLoopGroup,bossGroup 负责接受客户端连接,workerGroup 负责处理客户端的 I/O 操作EventLoopGroup bossGroup = new MultiThreadIoEventLoopGroup(1, NioIoHandler.newFactory());EventLoopGroup workerGroup = new MultiThreadIoEventLoopGroup(NioIoHandler.newFactory());try {ServerBootstrap bootstrap = new ServerBootstrap();bootstrap.group(bossGroup, workerGroup).channel(NioServerSocketChannel.class).childHandler(new ChannelInitializer<Channel>() {@Overrideprotected void initChannel(Channel ch) throws Exception {ch.pipeline().addLast(new HttpServerCodec());ch.pipeline().addLast(new HttpObjectAggregator(65536));ch.pipeline().addLast(new ChunkedWriteHandler());ch.pipeline().addLast(new WebSocketServerProtocolHandler("/websocket"));ch.pipeline().addLast(new WebSocketServerHandler());}});ChannelFuture future = bootstrap.bind(8080).sync();future.channel().closeFuture().sync();} finally {bossGroup.shutdownGracefully();workerGroup.shutdownGracefully();}}public static class WebSocketServerHandler extends SimpleChannelInboundHandler<Object> {@Overrideprotected void channelRead0(ChannelHandlerContext ctx, Object msg) throws Exception {if (msg instanceof FullHttpRequest) {System.out.println("处理 WebSocket 握手请求");// 处理 WebSocket 握手请求FullHttpRequest request = (FullHttpRequest) msg;if (request.method() == HttpMethod.GET) {WebSocketServerHandshakerFactory wsFactory = new WebSocketServerHandshakerFactory("ws://localhost:8080/websocket", null, false);WebSocketServerHandshaker handshake = wsFactory.newHandshaker(request);if (handshake != null) {handshake.handshake(ctx.channel(), request);} else {WebSocketServerHandshakerFactory.sendUnsupportedVersionResponse(ctx.channel());}}} else if (msg instanceof WebSocketFrame frame) {// 处理 WebSocket 消息if (frame instanceof TextWebSocketFrame) {String request = ((TextWebSocketFrame) frame).text();ctx.channel().writeAndFlush(new TextWebSocketFrame("Hello, " + request));}}}@Overridepublic void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {cause.printStackTrace();ctx.close();}@Overridepublic void channelActive(ChannelHandlerContext ctx) throws Exception {System.out.println( ctx);}}
}

3. WebSocket 客户端实现

客户端通过 WebSocket 协议与服务器建立连接,并进行双向通信。客户端可以使用 Java 中的 WebSocket 客户端库进行实现。

package org.example.websocket.client;import org.java_websocket.client.WebSocketClient;
import org.java_websocket.handshake.ServerHandshake;import java.net.URI;
import java.util.Scanner;/*** @author Administrator*/
public class WebSocketClientApp extends WebSocketClient {public WebSocketClientApp(URI serverURI) {super(serverURI);}@Overridepublic void onOpen(ServerHandshake serverHandshake) {System.out.println("Connected to server");send("Hello WebSocket Server");}@Overridepublic void onMessage(String message) {System.out.println("Received message: " + message);}@Overridepublic void onClose(int code, String reason, boolean remote) {System.out.println("Closed with exit code " + code + " and reason: " + reason);}@Overridepublic void onError(Exception ex) {ex.printStackTrace();}public static void main(String[] args) throws Exception {WebSocketClientApp client = new WebSocketClientApp(new URI("ws://localhost:8080/websocket"));client.connect();Scanner in = new Scanner(System.in);while(true){String str = in.next();if("exit".equals(str)){client.close();break;}else{System.out.println();client.send(str);}}}
}

四、总结

通过结合 NettyWebSocket,我们能够构建一个高性能、低延迟、支持双向实时通信的应用系统。Netty 提供了高效的网络通信能力,而 WebSocket 协议则满足了实时双向通信的需求。

使用这种方式,可以处理大量并发连接,同时提供快速的消息传递和响应能力。这对于需要高吞吐量和低延迟的应用场景,如在线游戏、实时聊天、金融交易等,具有显著优势。

希望本篇博客能帮助你理解如何使用 NettyWebSocket 搭建一个快速且稳定的双向通信通道。如果你对具体的实现或有任何问题,欢迎在评论区留言交流!

http://www.dtcms.com/a/364915.html

相关文章:

  • word文档中从某一页开始页码全是1
  • Wpf程序屏幕居中问题修复全记录
  • 39.Ansible: 包含与导入
  • FastVLM:高效视觉编码助力视觉语言模型突破高分辨率效率瓶颈
  • 独家|字节Seed部门增发百万期权,大模型战线开启“锁人”模式
  • 【golang长途旅行第37站】Redis连接池
  • MMD动画(一)模型、动作、音乐导入
  • 【大前端】React 父子组件通信、子父通信、以及兄弟(同级)组件通信
  • 科技赋能田园:数字化解决方案开启智慧农业新篇章
  • centos9 docker启动不起来,docker启动发生堵塞问题!
  • 【明道云】[工作表控件5] 手机控件的格式化处理
  • 【机器学习】实战:市场增长点分析挖掘项目
  • SyncBack 备份同步软件: 使用增量备份, 节省网络传输及存储成本
  • 【NVIDIA B200】2.all_reduce_perf NVIDIA B200 8-GPU 系统 All-Reduce 性能深度解析
  • 力扣115:不同的子序列
  • 热烈庆祝 | 一二三物联网携这款产品入选2025年度山东省首台(套)技术装备生产企业及产品名单
  • Day20 JavaScript 进阶核心:IIFE、代码规范、调试与对象模型
  • AI优化SEO关键词策略指南
  • 时序数据库选型指南:Apache IoTDB快速部署与实战应用
  • 时序数据库IoTDB的核心优势
  • mysql第五天学习 Mysql全局优化总结
  • 如何使文件夹内的软件或者文件不受windows 安全中心的监视
  • FastGPT社区版大语言模型知识库、Agent开源项目推荐
  • 智慧用电安全解决方案:打造全流程可视化管控
  • 知微集:Python中的线程Thread(一)
  • MCP是什么? 小白如何学习使用MCP?一篇文档带你详细了解神秘的MCP
  • 【C++】控制台输入与输出
  • osgb转b3dm转glb小工具流程-解决办法
  • Linux内核进程管理子系统有什么第四十一回 —— 进程主结构详解(37)
  • 《用 Flask 构建用户认证系统:从零开始实现注册与登录功能》