SpringBoot整合Netty实现TCP通信

SpringBoot整合Netty,实现TCP通信

  1. 组成部分为:SpringBoot + Mybatis + netty;

  2. 开发环境为Java8 ,IDEA,Maven;

  3. 开始教程

    1. 使用 idea 创建 SpringBoot 项目;

    2. 创建完成后在 pom.xml 文件 dependencies 写入以下代码:

       <!--netty-->
              <dependency>
                  <groupId>io.netty</groupId>
                  <artifactId>netty-all</artifactId>
                  <version>4.1.17.Final</version>
              </dependency>
      
    3. 创建三个 Netty 所需类

      1. NettyServer 为服务器监听器

        package cn.han.han_parse.service;
        
        import io.netty.bootstrap.ServerBootstrap;
        import io.netty.channel.ChannelFuture;
        import io.netty.channel.ChannelOption;
        import io.netty.channel.EventLoopGroup;
        import io.netty.channel.nio.NioEventLoopGroup;
        import io.netty.channel.socket.nio.NioServerSocketChannel;
        import lombok.extern.slf4j.Slf4j;
        
        import java.net.InetSocketAddress;
        
        /**
         * @author hanyiming
         * @create 2021/9/9 10:08
         */
        @Slf4j
        public class NettyServer {
            public void start() {
                System.out.println("这是Netty");
                InetSocketAddress socketAddress = new InetSocketAddress("127.0.0.1", 8080);
                //new 一个主线程组
                EventLoopGroup bossGroup = new NioEventLoopGroup(1);
                //new 一个工作线程组
                EventLoopGroup workGroup = new NioEventLoopGroup(200);
                ServerBootstrap bootstrap = new ServerBootstrap()
                        .group(bossGroup, workGroup)
                        .channel(NioServerSocketChannel.class)
                        .childHandler(new ServerChannelInitializer())
                        .localAddress(socketAddress)
                        //设置队列大小
                        .option(ChannelOption.SO_BACKLOG, 1024)
                        // 两小时内没有数据的通信时,TCP会自动发送一个活动探测数据报文
                        .childOption(ChannelOption.SO_KEEPALIVE, true);
                //绑定端口,开始接收进来的连接
                try {
                    ChannelFuture future = bootstrap.bind(socketAddress).sync();
                    log.info("服务器启动开始监听端口: {}", socketAddress.getPort());
                    future.channel().closeFuture().sync();
                } catch (InterruptedException e) {
                    log.error("服务器开启失败", e);
                } finally {
                    //关闭主线程组
                    bossGroup.shutdownGracefully();
                    //关闭工作线程组
                    workGroup.shutdownGracefully();
                }
            }
        }
        
        
      2. NettyServerHandler 为处理器

        • 注意在此处如果没写Controller层,需要在本类中添加@Component 注解,并初始化所需要的service层和mapper层;如果不需要可以删除,在以下代码中有@Component 注解;
        package cn.han.han_parse.service;
        
        import cn.han.han_parse.dao.BasicMapper;
        import cn.han.han_parse.entity.Basic;
        import cn.han.han_parse.util.ParseMessage;
        import io.netty.channel.ChannelHandlerContext;
        import io.netty.channel.ChannelInboundHandlerAdapter;
        import lombok.extern.slf4j.Slf4j;
        import org.springframework.beans.factory.annotation.Autowired;
        import org.springframework.stereotype.Component;
        
        import javax.annotation.PostConstruct;
        
        /**
         * @author hanyiming
         * @create 2021/9/9 10:12
         */
        @Slf4j
        @Component
        public class NettyServerHandler extends ChannelInboundHandlerAdapter {
        
            public static NettyServerHandler nettyServerHandler;
        
            @Autowired
            private BasicService basicService;
        
            @Autowired
            private BasicMapper basicMapper;
        
            @PostConstruct
            public void init() {
                nettyServerHandler = this;
                nettyServerHandler.basicService = this.basicService;
                nettyServerHandler.basicMapper = this.basicMapper;
                // 初使化时将已静态化的testService实例化
            }
        
            /**
             * 客户端连接会触发
             */
            @Override
            public void channelActive(ChannelHandlerContext ctx) throws Exception {
                log.info("连接成功...");
            }
        
            /**
             * 客户端发消息会触发
             */
            @Override
            public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception {
                log.info("服务器收到消息: {}", msg.toString());
                //此处为解析tcp参数所需要的方法!!!自定义
                /*final Basic parse = ParseMessage.parse(ctx, msg.toString());
                nettyServerHandler.basicService.addBasic(parse);*/
                ctx.write("你也好哦");
                ctx.flush();
            }
        
        
            /**
             * 发生异常触发
             */
            @Override
            public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {
                cause.printStackTrace();
                ctx.close();
            }
        }
        
        
      3. ServerChannelInitializer 服务初始化器,

        package cn.han.han_parse.service;
        
        import cn.han.han_parse.util.MyDecode;
        import io.netty.channel.ChannelInitializer;
        import io.netty.channel.socket.SocketChannel;
        import io.netty.handler.codec.string.StringDecoder;
        import io.netty.handler.codec.string.StringEncoder;
        import io.netty.util.CharsetUtil;
        import lombok.extern.slf4j.Slf4j;
        
        /**
         * @author hanyiming
         * @create 2021/9/9 10:10
         */
        @Slf4j
        public class ServerChannelInitializer extends ChannelInitializer<SocketChannel> {
        
            @Override
            protected void initChannel(SocketChannel socketChannel) throws Exception {
                //添加编解码,此处代码为解析tcp传过来的参数,为UTF-8格式,可以自定义解码格式
                socketChannel.pipeline().addLast("decoder", new StringDecoder(CharsetUtil.UTF_8));
                socketChannel.pipeline().addLast("encoder", new StringEncoder(CharsetUtil.UTF_8));
                socketChannel.pipeline().addLast(new NettyServerHandler());
            }
        }
        
        
      4. 此处为我所需的进制转换规则:

         /**
             * 十六进制转换为十进制
             *
             * @param content
             * @return
             */
            public static int covert(String content) {
                int number = 0;
                String[] HighLetter = {"A", "B", "C", "D", "E", "F"};
                Map<String, Integer> map = new HashMap<>();
                for (int i = 0; i <= 9; i++) {
                    map.put(i + "", i);
                }
                for (int j = 10; j < HighLetter.length + 10; j++) {
                    map.put(HighLetter[j - 10], j);
                }
                String[] str = new String[content.length()];
                for (int i = 0; i < str.length; i++) {
                    str[i] = content.substring(i, i + 1);
                }
                for (int i = 0; i < str.length; i++) {
                    number += map.get(str[i]) * Math.pow(16, str.length - 1 - i);
                }
                return number;
            }
        
            /**
             * 十进制转换为ASCII码
             *
             * @param ascii
             * @return
             */
            public static String dec2Str(String ascii) {
                StringBuilder sb = new StringBuilder();
                for (int i = 0; i < ascii.length() - 1; i += 2) {
                    String h = ascii.substring(i, (i + 2));
                    // 这里第二个参数传10表10进制
                    int decimal = Integer.parseInt(h, 10);
                    sb.append((char) decimal);
                }
                return sb.toString();
            }
        
  • 13
    点赞
  • 45
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
### 回答1: Spring Boot可以很方便地整合Netty实现TCP协议的通信。具体实现步骤如下: 1. 引入Netty依赖 在pom.xml文件中添加以下依赖: ``` <dependency> <groupId>io.netty</groupId> <artifactId>netty-all</artifactId> <version>4.1.25.Final</version> </dependency> ``` 2. 编写Netty服务端 编写一个Netty服务端,监听指定端口,接收客户端的请求,并返回响应。具体实现可以参考Netty官方文档。 3. 配置Spring Boot 在Spring Boot的配置文件中,配置Netty服务端的端口号和其他相关参数。 4. 启动Spring Boot应用程序 启动Spring Boot应用程序,Netty服务端会自动启动并监听指定端口。 5. 编写客户端程序 编写一个客户端程序,连接Netty服务端,并发送请求。具体实现可以参考Netty官方文档。 通过以上步骤,就可以实现Spring Boot整合Netty实现TCP协议的通信。 ### 回答2: Spring Boot是一个非常流行的Java开源框架,它提供了一种简单且快捷的方式来构建可扩展的Web应用程序。而Netty是一个基于NIO的客户端/服务器框架,它可以轻松处理高负载的网络通信。 因此通过Spring Boot和Netty整合,可以实现高效,快速,可扩展的TCP通信,在需要高性能可扩展的应用程序中是有很大的优势的。 实现过程如下: 1. 通过Spring Boot创建一个Maven项目,引入Netty依赖。 2. 创建Netty服务端和客户端,用于实现TCP通讯。服务端可以监听端口,客户端则可以连接服务端。 3. 将Netty的ChannelHandler封装成Spring Bean,并在Spring Boot中进行注入。 4. 通过使用Spring Boot的自动配置功能,将服务端和客户端的配置信息进行注入,从而使整个过程的配置更加简单。 5. 为了更好地支持多个客户端并发操作,可以使用Netty的线程池功能来提高性能和稳定性。 6. 配置Spring Boot,使其运行在指定的端口,并且注册Netty ChannelHandler,使其能够接收和处理来自客户端的请求消息。 7. 编写客户端代码,建立与服务端的连接并发送数据。 8. 客户端与服务端完成通信后,可以将数据响应给客户端,并断开连接。 通过以上步骤,就可以使用Spring Boot和Netty实现高效,快速,可扩展的TCP通信。这种架构有很多优点,例如高并发,高性能,易于维护,容易扩展等。对于需要实现实时数据传输和高性能的应用程序而言,这是一种非常好的解决方案。 ### 回答3: Springboot是一款非常流行的Java开发框架,它提供了很多便捷的工具和库,帮助开发者更快地搭建高效的应用程序。Netty则是一款基于NIO的高性能网络通信框架,非常适合开发高并发、高性能的网络应用。 利用Springboot整合Netty实现TCP通信可以方便地实现异步、非阻塞IO,而不需要开发者手动处理Java NIO的细节。下面简要介绍如何利用Springboot整合Netty实现TCP通信。 1. 引入Netty的依赖 在pom.xml文件中引入Netty的依赖,例如: ``` <dependency> <groupId>io.netty</groupId> <artifactId>netty-all</artifactId> <version>4.1.25.Final</version> </dependency> ``` 2. 实现Netty服务端 创建一个NettyServer类,继承自ChannelInboundHandlerAdapter,并实现以下方法: ``` public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception {} public void channelReadComplete(ChannelHandlerContext ctx) throws Exception {} public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {} ``` 在NettyServer类的构造方法中启动Netty服务端,示例代码如下: ``` public class NettyServer extends ChannelInboundHandlerAdapter { public NettyServer() { EventLoopGroup bossGroup = new NioEventLoopGroup(); EventLoopGroup workerGroup = new NioEventLoopGroup(); try { ServerBootstrap b = new ServerBootstrap(); b.group(bossGroup, workerGroup) .channel(NioServerSocketChannel.class) .option(ChannelOption.SO_BACKLOG, 128) .childOption(ChannelOption.SO_KEEPALIVE, true) .childHandler(new ChannelInitializer<SocketChannel>() { @Override public void initChannel(SocketChannel ch) throws Exception { ch.pipeline().addLast(new NettyServer()); } }); // Bind and start to accept incoming connections. ChannelFuture f = b.bind(PORT).sync(); // Wait until the server socket is closed. f.channel().closeFuture().sync(); } finally { workerGroup.shutdownGracefully(); bossGroup.shutdownGracefully(); } } @Override public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception { // 处理读事件 } @Override public void channelReadComplete(ChannelHandlerContext ctx) throws Exception { // 读事件完成处理 } @Override public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception { // 处理异常 } // 启动Netty服务端 public static void main(String[] args) { new NettyServer(); } } ``` 3. 实现Netty客户端 创建一个NettyClient类,继承自SimpleChannelInboundHandler,并实现以下方法: ``` public void channelActive(ChannelHandlerContext ctx) throws Exception {} protected void channelRead0(ChannelHandlerContext ctx, Object msg) throws Exception {} public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {} ``` 在NettyClient类的构造方法中启动Netty客户端,示例如下: ``` public class NettyClient extends SimpleChannelInboundHandler<String> { private final String host; private final int port; public NettyClient(String host, int port) { this.host = host; this.port = port; EventLoopGroup group = new NioEventLoopGroup(); try { Bootstrap b = new Bootstrap(); b.group(group) .channel(NioSocketChannel.class) .option(ChannelOption.TCP_NODELAY, true) .handler(new ChannelInitializer<SocketChannel>() { @Override public void initChannel(SocketChannel ch) throws Exception { ch.pipeline().addLast(new DelimiterBasedFrameDecoder(1024, Unpooled.copiedBuffer("$_".getBytes()))); ch.pipeline().addLast(new StringEncoder()); ch.pipeline().addLast(new StringDecoder()); ch.pipeline().addLast(new NettyClient(host, port)); } }); // Start the client. ChannelFuture f = b.connect(host, port).sync(); // Wait until the connection is closed. f.channel().closeFuture().sync(); } finally { group.shutdownGracefully(); } } @Override public void channelActive(ChannelHandlerContext ctx) throws Exception { // 发送消息 } @Override protected void channelRead0(ChannelHandlerContext ctx, String msg) throws Exception { // 处理读事件 } @Override public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception { // 处理异常 } // 启动Netty客户端 public static void main(String[] args) { String host = "127.0.0.1"; int port = 8080; new NettyClient(host, port); } } ``` 以上是利用Springboot整合Netty实现TCP通信的大致步骤。实际开发过程中还需要根据应用程序的具体需求进一步优化和调整。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值