我的编程空间,编程开发者的网络收藏夹
学习永远不晚

【异步】Java 的 8 种异步实现方式

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

【异步】Java 的 8 种异步实现方式

异步执行对于开发者来说并不陌生,在实际的开发过程中,很多场景多会使用到异步,相比同步执行,异步可以大大缩短请求链路耗时时间,比如:发送短信、邮件。

异步的八种实现方式:

  1. 线程异步 Thread/Runnable
  2. Future + Callable
  3. 异步框架 CompletableFuture
  4. Spring 注解 @Async
  5. Spring ApplicationEvent 事件
  6. 第三方异步框架,比如 Hutool 的 ThreadUtil
  7. Guava 异步
  8. 消息队列

1、线程异步

public class ThreadTest implements Runnable{    @Override    public void run() {        System.out.println(Thread.currentThread().getName());    }    public static void main(String[] args) {        ThreadTest threadTest = new ThreadTest();        new Thread(threadTest).start();    }}

当然,如果每次都创建一个 Thread 线程,频繁的创建、销毁,浪费系统资源,我们可以采用线程池:【Thread】线程池的 7 种创建方式及自定义线程池

2、Future 异步

public class FutureTest {    public static void main(String[] args) throws ExecutionException, InterruptedException {        ExecutorService executor = Executors.newFixedThreadPool(1);        Future<String> future = executor.submit(() -> {            Thread.sleep(2000);            return "this is future execute final result!!!";        });        //这里需要返回值时会阻塞主线程        String result = future.get();        System.out.println(result);        executor.shutdown();    }}

Future的不足之处的包括以下几点:

  1. 无法被动接收异步任务的计算结果:虽然我们可以主动将异步任务提交给线程池中的线程来执行,但是待异步任务执行结束之后,主线程无法得到任务完成与否的通知,它需要通过get方法主动获取任务执行的结果。
  2. Future件彼此孤立:有时某一个耗时很长的异步任务执行结束之后,你想利用它返回的结果再做进一步的运算,该运算也会是一个异步任务,两者之间的关系需要程序开发人员手动进行绑定赋予,Future并不能将其形成一个任务流(pipeline),每一个Future都是彼此之间都是孤立的,所以才有了后面的CompletableFuture,CompletableFuture就可以将多个Future串联起来形成任务流。
  3. Futrue没有很好的错误处理机制:截止目前,如果某个异步任务在执行发的过程中发生了异常,调用者无法被动感知,必须通过捕获get方法的异常才知晓异步任务执行是否出现了错误,从而在做进一步的判断处理

3、CompletableFuture
关于 CompletableFuture 更多详情请看:【异步】Futurn、FutureTask、CompletionService、CompletableFuture

public static void thenRunAsync() throws ExecutionException, InterruptedException {    CompletableFuture<Integer> cf1 = CompletableFuture.supplyAsync(() -> {        System.out.println(Thread.currentThread() + " cf1 do something....");        return 1;    });    CompletableFuture<Void> cf2 = cf1.thenRunAsync(() -> {        System.out.println(Thread.currentThread() + " cf2 do something...");    });    //等待任务1执行完成    System.out.println("cf1结果->" + cf1.get());    //等待任务2执行完成    System.out.println("cf2结果->" + cf2.get());}

4、Spring 注解 @Async

@Configurationpublic class ThreadPoolConfig {    @Bean("taskExecutor")    public Executor taskExecutor() {        //返回可用处理器的Java虚拟机的数量 12        int i = Runtime.getRuntime().availableProcessors();        System.out.println("系统最大线程数  : " + i);        ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();        //核心线程池大小        executor.setCorePoolSize(16);        //最大线程数        executor.setMaxPoolSize(20);        //配置队列容量,默认值为Integer.MAX_VALUE        executor.setQueueCapacity(99999);        //活跃时间        executor.setKeepAliveSeconds(60);        //线程名字前缀        executor.setThreadNamePrefix("asyncServiceExecutor -");        //设置此执行程序应该在关闭时阻止的最大秒数,以便在容器的其余部分继续关闭之前等待剩余的任务完成他们的执行        executor.setAwaitTerminationSeconds(60);        //等待所有的任务结束后再关闭线程池        executor.setWaitForTasksToCompleteOnShutdown(true);        return executor;    }}
@Service@EnableAsyncpublic class AsyncServiceImpl implements AsyncService {    @Override    @Async("taskExecutor")    public String sendSms() {        System.out.println(Thread.currentThread().getName());        return null;    }    @Override    @Async("taskExecutor")    public String sendEmail() {        System.out.println(Thread.currentThread().getName());        return null;    }}

在实际项目中, 使用 @Async 调用线程池,推荐等方式是是使用自定义线程池的模式,不推荐直接使用 @Async 直接实现异步

5、Spring ApplicationEvent 事件

Spring 中使用事件只需要以下的几个步骤:

  1. 定义事件,继承 ApplicationEvent
  2. 定义监听,要么实现 ApplicationListener 接口,要么在方法上添加 @EventListener 注解
  3. 定义发布事件接口,调用 ApplicationContext.publishEvent() 或者 ApplicationEventPublisher.publishEvent();
  4. 业务调用发布事件
@Getter@Setterpublic class BaseEvent<T> extends ApplicationEvent {    private T data;    public BaseEvent(Object source) {        super(source);    }    public BaseEvent(Object source, T data) {        super(source);        this.data = data;    }}
@Componentpublic class BaseEventListener implements ApplicationListener<BaseEvent<UserVo>> {    @Override    @Async("taskExecutor")    public void onApplicationEvent(BaseEvent<UserVo> baseEvent) {        UserVo eventData = baseEvent.getData();        // TODO 业务处理    }}
@Autowiredprivate ApplicationContext applicationContext;@GetMapping("/pubEvent")public void pubEvent() {    BaseEvent<UserVo> baseEvent = new BaseEvent<>("event", new UserVo());    applicationContext.publishEvent(baseEvent);}

6、Hutool 的 ThreadUtil

public static void main(String[] args) {    for (int i = 0; i < 3; i++) {        ThreadUtil.execAsync(() -> {            ThreadLocalRandom threadLocalRandom = ThreadLocalRandom.current();            int number = threadLocalRandom.nextInt(20) + 1;            System.out.println(number);        });        log.info("当前第:" + i + "个线程");    }    log.info("task finish!");}

7、 Guava 异步

public static void test() {    ListeningExecutorService executorService = MoreExecutors.listeningDecorator(Executors.newCachedThreadPool());    final ListenableFuture<Integer> listenableFuture = executorService.submit(() -> {        log.info("callable execute...");        TimeUnit.SECONDS.sleep(1);        return 1;    });    Futures.addCallback(listenableFuture, new FutureCallback<Integer>() {        @Override        public void onSuccess(@Nullable Integer integer) {            System.out.println("Get listenable future's result with callback " + integer);        }        @Override        public void onFailure(Throwable throwable) {            throwable.printStackTrace();        }    }, Executors.newCachedThreadPool());}

8、 消息队列

常用的消息队列:RabbitMqRocketMq

来源地址:https://blog.csdn.net/sco5282/article/details/131112037

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

【异步】Java 的 8 种异步实现方式

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

java异步编程的7种实现方式小结

异步处理的实现方式有很多种,常见多线程,消息中间件,发布订阅的广播模式,本文就详细的介绍java异步编程的7种实现方式,感兴趣的可以了解一下
2023-03-23

java 实现开启异步线程的几种方式

在Java中,有多种方式可以实现异步线程以避免在主线程中执行耗时操作导致界面卡顿的问题。以下是几种常用的方式: 使用Thread类:可以使用Thread类来创建一个新的线程,并在其run()方法中执行耗时操作。例如: new Thread(
2023-08-17

web前端:解决异步的几种实现方式

编程学习网: 异步双方不需要共同的时钟,也就是接收方不知道发送方什么时候发送,所以在发送的信息中就要有提示接收方开始接收的信息,如开始位,同时在结束时有停止位。
web前端:解决异步的几种实现方式
2024-04-23

java异步编程的实现方式有哪些

这篇文章主要介绍“java异步编程的实现方式有哪些”,在日常操作中,相信很多人在java异步编程的实现方式有哪些问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”java异步编程的实现方式有哪些”的疑惑有所帮助!
2023-07-05

Android异步更新UI的四种方式

大家都知道由于性能要求,android要求只能在UI线程中更新UI,要想在其他线程中更新UI,大致有4种方式,下面分别使用四种方式来更新一个TextView。 1.使用Handler消息传递机制package com.example.run
2022-06-06

C#BeginInvoke实现异步编程方式

这篇文章主要介绍了C#BeginInvoke实现异步编程方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
2023-01-28

java实现异步的方法有哪些

Java实现异步的方法有以下几种:1. 使用线程池:可以通过ExecutorService接口创建线程池,然后使用submit()或execute()方法提交任务进行异步执行。2. 使用CompletableFuture:Completab
2023-08-09

javascript实现异步的方法

这篇文章给大家分享的是有关javascript实现异步的方法的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。javascript实现异步的方法:1、使用setTimeout方法;2、使用setImmediate 方
2023-06-14

C#实现异步的常用方式总结

这篇文章主要为大家详细介绍了C#实现异步的几个常用方式,文中的示例代码讲解详细,具有一定的学习价值,感兴趣的小伙伴可以了解一下
2023-05-19

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录