ThreadPoolTaskExecutor使用

ThreadPoolTaskExecutor使用使用场景:多线程

大家好,又见面了,我是你们的朋友全栈君。

使用场景:多线程消费

1.异步注解@EnableAsync配置 

2.配置线程池bean(启动类者单独配置bean),可参考本菜鸟的其他文章

3.启动消费者;生产者可以通过其他方式

4.实际业务处理,配置异步处理注解和线程池名称 @Async(“redisThread”)

步骤一:配置异步注解和线程池

@SpringBootApplication
@EnableAsync
public class ImRedisApplication {
    public static void main(String[] args) {
        SpringApplication.run(ImRedisApplication.class, args);
    }

      @Bean("redisThread")
    public ThreadPoolTaskExecutor initRedisPool(){
        ThreadPoolTaskExecutor threadPoolTaskExecutor = new ThreadPoolTaskExecutor();
        /**
         * 核心线程数
         */
        threadPoolTaskExecutor.setCorePoolSize(4);
        /**
         * 最大线程数
         */
        threadPoolTaskExecutor.setMaxPoolSize(9);
        /**
         * 存活
         */
        threadPoolTaskExecutor.setKeepAliveSeconds(10 * 60);
        /**
         * 队列容量
         */
        threadPoolTaskExecutor.setQueueCapacity(10);
        /**
         * 线程名称
         */
        threadPoolTaskExecutor.setThreadNamePrefix("redisThread" + "-");
        /**
         * 拒绝策略,关于拒绝策略可以看实现 RejectedExecutionHandler 的类,当然也可以自定义拒接策略,也就是现实RejectedExecutionHandler接口方法
         */
        threadPoolTaskExecutor.setRejectedExecutionHandler(new ThreadPoolExecutor.AbortPolicy());
        return threadPoolTaskExecutor;
    }
}

2.通过 @PostConstruct和线程启动消费者,(生产者可以使用xxl-job,MQ或者其他方式产生,基于自己的实际业务需求)

@Component
@Slf4j
public class RedisConsumerInit {

    @Autowired
    private IdConsumerService idConsumerService;

    @PostConstruct
    public void init() {
        /**
         * 初始化启动
         */
        new PoolConsumerThread().start();
    }

    class PoolConsumerThread extends Thread {
        @Override
        public void run() {

            while (true) {
                /**
                 * 从redis,MQ等拿消息
                 */
                int i = new Random().nextInt();
                try {
                    idConsumerService.execute(i);
                } catch (RejectedExecutionException e) {
                    log.error("异常了:{}", e.getMessage());
                    /**
                     * 如遇到线程池拒绝抛异常,做其他业务处理,比如继续放置到redis队列处理或者其他
                     */
                } catch (Exception e) {
                    log.error("异常了:{}", e.getMessage());
                }

                try {
                    TimeUnit.NANOSECONDS.sleep(1);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

3.具体业务处理,特别注意开启异步注解并配置线程池的bean

@Service
@Slf4j
public class IdConsumerService {


    @Async("redisThread")
    public void execute(Integer num){
        try {
            log.info("当前消费:{}", num);
            /**
             * 实际业务处理
             */

        } catch (Exception e) {
            e.printStackTrace();
        }
    }
    
}

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/149658.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 小心onpropertychange在ie8上的地雷[通俗易懂]

    onpropertychangeonpropertychange的话,只要当前对象属性发生改变,都会触发事件,但是它是IE专属的

    2022年4月10日
    34
  • a4纸两版的小册子怎么打印_word怎么打印小册子用a4

    a4纸两版的小册子怎么打印_word怎么打印小册子用a4用word打印小册子具体操作步骤:1、首先打开将要打印的文档,然后在菜单栏下面的快捷菜单栏,找到打印机的快捷图标,单击鼠标左键选择打开打印机。2、会弹出打印机设置对话框,可以进行打印前的一些细节的设置,由于是要选择打印一本小册子所以选择打印“所有页面”。3、在下面一栏是调整页面大小,也就是打印模式,选择“小册子”进行打印。4、然后下面是关于小册子的一些设置,按照自己的需要进行设置。5、设置完成以后…

    2022年9月6日
    3
  • Navicat Premium 15激活码(最新序列号破解)

    Navicat Premium 15激活码(最新序列号破解),https://javaforall.net/100143.html。详细ieda激活码不妨到全栈程序员必看教程网一起来了解一下吧!

    2022年3月18日
    708
  • SpringMVC 工作原理

    SpringMVC 工作原理1.客户端请求提交到DispatcherServlet2.由DispatcherServlet控制器查询一个或多个HandlerMapping,找到处理请求的Controller3.DispatcherServlet将请求提交到Controller4.Controller调用业务逻辑处理后,返回ModelAndView5.DispatcherServlet查询一个或多个ViewResoler视图解析器,找到ModelAndView指定的视图,并将结果显示到客户

    2022年5月14日
    37
  • Vue高阶组件_高阶组件的承上启下

    Vue高阶组件_高阶组件的承上启下目录一、高阶组件概念二、目标三、思路四、准备五、实现六、难点Reference一、高阶组件概念何谓高阶组件?类比高阶函数的定义:将函数作为参数的函数就是高阶函数,那么,将组件作为参数的组件就是高阶组件。二、目标假如我们有一个组件,我们希望通过某个函数,去扩展它,得到一个新的组件,新的组件有完全的参数组件的行为,如果这点可以满足,那么其他扩展就可以针对性的…

    2022年10月25日
    0
  • 计算机编程原理_如何编程

    计算机编程原理_如何编程一、Socket简介Socket是进程通讯的一种方式,即调用这个网络库的一些API函数实现分布在不同主机的相关进程之间的数据交换。几个定义:(1)IP地址:即依照TCP/IP协议分配给本地主机的网络地址,两个进程要通讯,任一进程首先要知道通讯对方的位置,即对方的IP。(2)端口号:用来辨别本地通讯进程,一个本地的进程在通讯时均会占用一个端口号,不同的进程端口号不同,因此在通

    2022年10月17日
    0

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号