近期做一些物联网方面项目,使用到了tcp协议,之前公司做过socket短连接,网上找了一个简单的demo,很早便学习到nio方面知识,学习了《netty从入门到精通》这本书,同时也根据网上视频做了几个demo,但学习不太深入,刚好物联网项目,就直接使用netty,前期直接使用这个框架,但作为近十年开发的java程序员,没有spring感觉总少了很多不便,后期改框架时想到了使用springboot整合netty代码。
先说之前项目中做socket代码怎么用spring管理的:利用springboot启动后启动socket服务,是没有被spring管理的
public class CouponApplication { public static void main(String[] args) { log.info("当前版本为:couponmisserver-0.0.5-SNAPSHOT"); log.info("版本日期:2020-1-7"); ApplicationContext context = SpringApplication.run(CouponApplication.class, args); //启动stock服务 Integer port = Integer.valueOf(Config.getConfig().get("socketPort")); SocketServer socketServer = new SocketServer(); socketServer.init(port, context); }}
网上一大神推荐的一个使用`CommandLineRunner `来加一个线程启动netty
@SpringBootApplication@Slf4j@EnableSchedulingpublic class Application implements CommandLineRunner { @Autowired private NettyServer nettyServer; public static void main(String[] args) { SpringApplication.run(Application.class, args); } @Override public void run(String... args) throws Exception { nettyServer.serverRun(); }}
我目前也是这样做的,这样netty就被整合到springboot中去了,但是近期chatGPT非常火,晚上无聊时我也拿它来玩时问到它这个问题,看这个大神怎么回答的:
Spring Boot 中可以通过创建自定义的
NettyServer
或NettyClient
类,并在其中实现 Netty 的服务端或客户端代码,然后在 Spring Boot 的应用程序中将其作为一个 Bean 注入到 Spring 容器中。以下是一个简单的示例代码,演示了如何整合 Netty 到 Spring Boot 应用程序中。首先,创建一个自定义的
NettyServer
类,用于实现 Netty 的服务端代码:
@Componentpublic class NettyServer { @Value("${netty.server.port}") private int port; @PostConstruct public void start() throws InterruptedException { EventLoopGroup bossGroup = new NioEventLoopGroup(); EventLoopGroup workerGroup = new NioEventLoopGroup(); try { ServerBootstrap serverBootstrap = new ServerBootstrap(); serverBootstrap.group(bossGroup, workerGroup) .channel(NioServerSocketChannel.class) .childHandler(new ChannelInitializer() { @Override protected void initChannel(SocketChannel ch) throws Exception {ChannelPipeline pipeline = ch.pipeline();pipeline.addLast(new StringDecoder());pipeline.addLast(new StringEncoder());pipeline.addLast(new NettyServerHandler()); } }); ChannelFuture channelFuture = serverBootstrap.bind(port).sync(); channelFuture.channel().closeFuture().sync(); } finally { bossGroup.shutdownGracefully(); workerGroup.shutdownGracefully(); } }}
之前网上那个大神给我的建议,我在操作过程中junit测试时总要先把启动netty那个给注释掉,无疑chatGPT大神给我找到了另一种解决方案,而且比我用的要好的多,我直接使用其给的方案重新优化了代码。
一、启动类不要动,还是原来的springboot启动类,在 NettyServer
上使用@Component注解让spring扫瞄管理,使用@PostConstruct注解直接运行netty服务
@Component@Slf4jpublic class NettyServer { @Autowired private NettyServerInitializer nettyServerInitializer; @Value("${nettyPort}") private Integer nettyPort; @PostConstruct public void serverRun() { //循环组接收连接,不进行处理,转交给下面的线程组 EventLoopGroup bossGroup = new NioEventLoopGroup(); //循环组处理连接,获取参数,进行工作处理 EventLoopGroup workerGroup = new NioEventLoopGroup(); try { //服务端进行启动类 ServerBootstrap serverBootstrap = new ServerBootstrap(); //使用NIO模式,初始化器等等 serverBootstrap.group(bossGroup, workerGroup).channel(NioServerSocketChannel.class).childHandler(nettyServerInitializer); //绑定端口 ChannelFuture channelFuture = serverBootstrap.bind(nettyPort).sync(); log.info("tcp服务器已经启动…………"); channelFuture.channel().closeFuture().sync(); } catch (InterruptedException e) { e.printStackTrace(); } finally { bossGroup.shutdownGracefully(); workerGroup.shutdownGracefully(); } }}
二、添加NettyServerInitializer类加上@Component注解让spring扫瞄管理,使用@Autowired进行依赖注入管理
@Componentpublic class NettyServerInitializer extends ChannelInitializer { Logger log = LogManager.getLogger(NettyServerInitializer.class); @Autowired private NettyHandler nettyHandler; //连接注册,创建成功,会被调用 @Override protected void initChannel(SocketChannel ch) throws Exception { log.info("==================netty报告=================="); log.info("信息:有一客户端链接到本服务端"); log.info("IP:{}", ch.remoteAddress().getAddress()); log.info("Port:{}", ch.remoteAddress().getPort()); log.info("通道id:{}", ch.id().toString()); log.info("==================netty报告完毕=================="); ChannelPipeline pipeline = ch.pipeline(); //定义读写空闲时间——(单位秒) pipeline.addLast(new IdleStateHandler(180, 60,180)); //注册拦截器 pipeline.addLast(nettyHandler); }}
三、其它NettyServerHandler类加上@Component注解让spring扫瞄管理,使用@Autowired进行依赖注入管理,但注意需要添加@ChannelHandler.Sharable注解,防止程序报错
@Component@ChannelHandler.Sharable@Slf4jpublic class NettyHandler extends SimpleChannelInboundHandler { @Autowired private DtuRegister dtuRegister; @Override protected void channelRead0(ChannelHandlerContext ctx, ByteBuf msg) throws Exception { //读到的客户端信息的逻辑处理(略) } @Override public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) { cause.getCause(); NettyServiceCommon.deleteKey(ctx.channel()); ctx.channel().close(); }}
至此,springboot整合netty完美解决,这才是便于理解和操作的方式,感谢网上使用监听器启动netty服务的那位大神,也感谢chatGPT大神给我提供另一种好的思路和方法
来源地址:https://blog.csdn.net/wskvfhprrk/article/details/129154410