SpringBoot多线程异步任务:ThreadPoolTaskExecutor + CompletableFuture

这篇具有很好参考价值的文章主要介绍了SpringBoot多线程异步任务:ThreadPoolTaskExecutor + CompletableFuture。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

1. SpringBoot多线程异步任务

1.1. 需求

SpringBoot 项目中,一个任务比较复杂,执行时间比较长,需要采用 多线程异步 的方式执行,从而缩短任务执行时间。

1.2. 多线程异步

  1. 将任务拆分成多个独立的子任务,每个子任务在独立子线程中执行;
  2. 当所有子任务的子线程全部执行完成后,将几个子任务汇总,得到总任务的执行结果。
  • 非阻塞:在汇总子任务时,不会阻塞主线程,也就是说汇总任务,也是在子线程执行的。开启了执行子任务和执行汇总任务的线程后,主线程就继续向下执行了。
  • 阻塞:在汇总子任务时,阻塞主线程,等待所有子任务执行完成并且汇总后,程序才继续向下执行。

2. 解决方案

ThreadPoolTaskExecutor + CompletableFuture

ThreadPoolTaskExecutor:是Spring框架提供的。
CompletableFuture:是 java 提供的(Java8 及以上)。

3. 代码实现

3.1. 异步任务和同步任务

package com.example.async.service;

import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ExecutionException;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;
import org.springframework.stereotype.Service;

@Service
public class TaskService {

	@Autowired
	private ThreadPoolTaskExecutor executor;

	public void executeAsyncNonBlocking() {
		System.out.println("----------执行异步任务(非阻塞),开始----------");

		long start = System.currentTimeMillis();
		CompletableFuture<Void> task1Future = CompletableFuture.runAsync(() -> {
			task1();
		}, executor);

		CompletableFuture<Void> task2Future = CompletableFuture.runAsync(() -> {
			task2();
		}, executor);

		CompletableFuture<Void> task3Future = CompletableFuture.runAsync(() -> {
			task3();
		}, executor);

		CompletableFuture.allOf(task1Future, task2Future, task3Future).thenRun(() -> {
			System.out.println("----------执行所有异步任务,结束----------");
			System.out.println("执行时间:" + (System.currentTimeMillis() - start) + " 毫秒");
		});

		System.out.println("----------执行异步任务(主线程),结束----------");
		System.out.println("执行时间:" + (System.currentTimeMillis() - start) + " 毫秒");
	}

	/**
	 * 执行异步任务
	 * 
	 * @throws ExecutionException
	 * @throws InterruptedException
	 */
	public void executeAsync() throws InterruptedException, ExecutionException {
		System.out.println("----------执行异步任务,开始----------");

		long start = System.currentTimeMillis();
		CompletableFuture<Void> task1Future = CompletableFuture.runAsync(() -> {
			task1();
		}, executor);

		CompletableFuture<Void> task2Future = CompletableFuture.runAsync(() -> {
			task2();
		}, executor);

		CompletableFuture<Void> task3Future = CompletableFuture.runAsync(() -> {
			task3();
		}, executor);

		CompletableFuture.allOf(task1Future, task2Future, task3Future).get();

		long end = System.currentTimeMillis();

		System.out.println("----------执行异步任务,结束----------");
		System.out.println("执行时间:" + (end - start) + " 毫秒");
	}

	/**
	 * 执行同步任务
	 */
	public void executeSync() {
		System.out.println("----------执行同步任务,开始----------");

		long start = System.currentTimeMillis();
		task1();
		task2();
		task3();
		long end = System.currentTimeMillis();

		System.out.println("----------执行同步任务,结束----------");
		System.out.println("执行时间:" + (end - start) + " 毫秒");
	}

	private void task1() {
		try {
			System.out.println("task1 开始");
			long start = System.currentTimeMillis();
			Thread.sleep(1000);
			long end = System.currentTimeMillis();
			System.out.println("task1 结束,执行时间:" + (end - start) + " 毫秒");
		} catch (InterruptedException e) {
			throw new RuntimeException(e);
		}
	}

	private void task2() {
		try {
			System.out.println("task2 开始");
			long start = System.currentTimeMillis();
			Thread.sleep(2000);
			long end = System.currentTimeMillis();
			System.out.println("task2 结束,执行时间:" + (end - start) + " 毫秒");
		} catch (InterruptedException e) {
			throw new RuntimeException(e);
		}
	}

	private void task3() {
		try {
			System.out.println("task3 开始");
			long start = System.currentTimeMillis();
			Thread.sleep(3000);
			long end = System.currentTimeMillis();
			System.out.println("task3 结束,执行时间:" + (end - start) + " 毫秒");
		} catch (InterruptedException e) {
			throw new RuntimeException(e);
		}
	}

}

3.2. 执行任务(异步任务和同步任务)

package com.example.async.controller;

import java.util.concurrent.ExecutionException;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import com.example.async.service.TaskService;

@RestController
@RequestMapping("test")
public class TaskController {

	@Autowired
	private TaskService service;

	@GetMapping("asyncNonBlocking")
	public String asyncNonBlocking() {
		// 异步执行任务
		service.executeAsyncNonBlocking();
		return "异步任务(非阻塞)已完成!";
	}

	@GetMapping("async")
	public String async() {
		// 异步执行任务
		try {
			service.executeAsync();
		} catch (InterruptedException e) {
			e.printStackTrace();
		} catch (ExecutionException e) {
			e.printStackTrace();
		}
		return "异步任务已完成!";
	}

	@GetMapping("sync")
	public String sync() {
		// 同步执行任务
		service.executeSync();
		return "同步任务已完成!";
	}

}

4. 执行结果

4.1. 同步执行任务的时间

执行同步任务的方法 executeSync,执行的结果和执行所需的时间如下:
SpringBoot多线程异步任务:ThreadPoolTaskExecutor + CompletableFuture

4.2. 异步执行任务的时间

执行异步任务的方法 executeAsync,执行的结果和执行所需的时间如下:
SpringBoot多线程异步任务:ThreadPoolTaskExecutor + CompletableFuture

4.3. 异步执行任务(非阻塞)的时间

执行异步任务(非阻塞)的方法 executeAsyncNonBlocking,执行的结果和执行所需的时间如下:
SpringBoot多线程异步任务:ThreadPoolTaskExecutor + CompletableFuture
主线程执行结束,接口就直接返回响应给用户了,任务在后台继续执行,直到所有任务全部执行完成。

4.4. 结论

  1. 异步任务,可以将几个子任务同时执行,然后在总任务中汇总。
  2. 异步任务总的执行时间,就是多个子任务执行时间的最大值。
  3. 非阻塞任务,接口会直接返回响应,任务仍在后台执行直至完成。

5. 示例项目(Gitee开源)

异步任务示例项目(Gitee开源)文章来源地址https://www.toymoban.com/news/detail-498317.html

到了这里,关于SpringBoot多线程异步任务:ThreadPoolTaskExecutor + CompletableFuture的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • SpringBoot整合、SpringBoot与异步任务

    java 的代码是同步顺序执行,当我们需要执行异步操作时我们通常会去创建一个新线程去执行。比如new Thread()。start(),或者使用线程池线程池 new ThreadPoolExecutor().execute 。 在 Springboot 中对其进行了简化处理,使用@EnableAsync + @Async 可以快速开启一个异步线程执行任务 启动类上使

    2024年02月13日
    浏览(35)
  • SpringBoot 实现异步任务

    在多数的Java项目中,在很多的场景都是用同步的方式去实现模块间的相互调用,在模块调用间可能会造成一些延迟,本篇文章将使用SpringBoot 去实现异步之间的调用, 提高系统的并发性能、用户体验。 简单直观:同步任务的执行是顺序的,代码执行的流程清晰明了,易于理

    2024年02月10日
    浏览(36)
  • SpringBoot 异步、邮件任务

    创建一个Hello项目 创建一个类AsyncService 异步处理还是非常常用的,比如我们在网站上发送邮件,后台会去发送邮件,此时前台会造成响应不动,直到邮件发送完毕,响应才会成功,所以我们一般会采用多线程的方式去处理这些任务。 编写方法,假装正在处理数据,使用线程

    2024年02月13日
    浏览(46)
  • SpringBoot(19)异步任务

    有时候,前端可能提交了一个耗时任务,如果后端接收到请求后,直接执行该耗时任务,那么前端需要等待很久一段时间才能接受到响应。如果该耗时任务是通过浏览器直接进行请求,那么浏览器页面会一直处于转圈等待状态。 事实上,当后端要处理一个耗时任务时,通常都

    2024年02月16日
    浏览(33)
  • SpringBoot(十三)异步任务

    目录 异步任务 1.1 什么叫异步 1、Java线程处理 2、SpringBoot异步任务 2.1 使用注解@EnableAsync开启异步任务支持 2.2、使用@Async注解标记要进行异步执行的方法 2.3、controller测试 3、异步任务相关限制 4、自定义 Executor(自定义线程池) 4.1、应用层级: 4.2、方法层级: 有时候,前端

    2024年02月04日
    浏览(33)
  • SpringBoot异步任务获取HttpServletRequest

    在使用框架日常开发中需要在controller中进行一些异步操作减少请求时间,但是发现在使用@Anysc注解后会出现Request对象无法获取的情况,本文就此情况给出完整的解决方案 @Anysc注解会开启一个新的线程,主线程的Request和子线程是不共享的,所以获取为null 在使用springboot的自定

    2024年02月21日
    浏览(51)
  • SpringBoot异步任务及并行事务实现

            上一篇介绍了原生Java如何实现串行/并行任务,主要使用了线程池 + Future + CountDownLatch,让主线程等待子线程返回后再向下进行。而在SpringBoot中,利用@Async和AOP对异步任务提供了更加便捷的支持,下面就针对SpringBoot使用异步任务需要注意的细节做一些分析。      

    2024年02月02日
    浏览(43)
  • SpringBoot原理分析 | 任务:异步、邮件、定时

    💗wei_shuo的个人主页 💫wei_shuo的学习社区 🌐Hello World ! 异步任务 Java异步指的是在程序执行过程中,某些任务可以在后台进行,而不会阻塞程序的执行。通常情况下,Java异步使用线程池来实现,将任务放入线程池中,等待线程池中的线程执行这些任务。Java异步可以提高程

    2024年02月16日
    浏览(46)
  • spring中使用ThreadPoolTaskExecutor配置线程池

    背景 spring中经常使用 ThreadPoolTaskExecutor 来调用JDK的ThreadPoolExecutor初始化线程池, 尤其在有异步执行的任务时, 由于spring 异步任务默认使用的executor 不会reuse线程 , 因此需要设置默认线程池, 来 替换 spring 默认的 SimpleAsyncTaskExecutor 参数 ThreadPoolTaskExecutor 不同的参数设置会使exec

    2024年02月01日
    浏览(29)
  • Spring(19) ThreadPoolTaskExecutor 线程池的使用

    1.1 为什么使用线程池 降低系统资源消耗: 通过重用已存在的线程,降低线程创建和销毁造成的消耗; 提高系统响应速度: 当有任务到达时,通过复用已存在的线程,无需等待新线程的创建便能立即执行; 方便线程并发数的管控: 因为线程若是无限制的创建,可能会导致内

    2024年01月19日
    浏览(35)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包