如题:当使用springBoot的ApplicationRunner初始化时启动使用netty做的udp两个端口的服务,解决执行完第一个服务之后,后面所有的方法都不执行的问题


  • 首先看一下我改动之前的代码,我写了两个服务器,说白了就是开启两个不同的端口,来监听不同客户端发来的数据
  • springboot 集成pulsar springboot 集成netty 启动等待问题_java


  • 之后写一个springBoot启动类,我这里实现的是ApplicationRunner,端口8822和8833都有里面有打印输入,然后我们启动项目
  • springboot 集成pulsar springboot 集成netty 启动等待问题_springboot 集成pulsar_02


  • 可以看到只执行了第一个8822端口的服务,下面所有的方法都没有执行,当然如果你只是启动一个服务,那只需要把启动服务的start()方法放到最下面执行也是可以的(原因是:我们在启动netty时会执行channelFuture.channel().closeFuture().sync(),Netty 会进入无限循环之中,阻塞了主线程,将会不再加载和扫描之后的类了。有可能你还需要 mybatis、redis 等,可能就扫描不到了
  • springboot 集成pulsar springboot 集成netty 启动等待问题_springboot 集成pulsar_03

  • 解决方法:
  • 采用异步方式使用线程池来启动服务
  • 使用@Async注解,对这个注解不熟悉的可以自行百度查阅,方法加上该注解可以使其异步执行,但有个问题就是使用该注解时如果不指定线程池的名称,则使用Spring默认的线程池,Spring默认的线程池为SimpleAsyncTaskExecutor。 它有一个缺陷那就是:并发情况下,会无限创建线程。。。
  • 所以我们这边要自定义线程池的配置,代码如下:
@Configuration
public class SpringAsyncConfig {

    @Bean("MyExecutor")//这里我指定了线程池的名称
    public Executor asyncServiceExecutor(){
        ThreadPoolTaskExecutor threadPoolTaskExecutor = new ThreadPoolTaskExecutor();
        //设核心线程
        threadPoolTaskExecutor.setCorePoolSize(10);
        //最大线程数
        threadPoolTaskExecutor.setMaxPoolSize(100);
        //缓冲队列大小
        threadPoolTaskExecutor.setQueueCapacity(10);
        //等待任务在关机时完成--表明等待所有线程执行完
        threadPoolTaskExecutor.setWaitForTasksToCompleteOnShutdown(true);
        //等待时间
        threadPoolTaskExecutor.setAwaitTerminationSeconds(60);
        //拒绝策略走默认
        //线程名称前缀
        threadPoolTaskExecutor.setThreadNamePrefix("MyAsync-");
        // 初始化线程
        threadPoolTaskExecutor.initialize();
        return threadPoolTaskExecutor;
    }

}
  • 然后回到NettyUDPServer服务类种加上注解 @Async("MyExecutor")括号里面指定刚刚上面定义的名称即可,加到方法上,如下图:

springboot 集成pulsar springboot 集成netty 启动等待问题_mybatis_04

  • 现在启动项目来看下执行结果是不是我们想要的:

springboot 集成pulsar springboot 集成netty 启动等待问题_springboot 集成pulsar_05


ok!至此完成,成功解决项目启动时初始化netty开启两个服务其中一个无法启动的问题,如有问题欢迎支持。