Callable与Futrue.get()等待线程执行完毕

1. 等待一个子线程执行完毕

public static void main(String[] args) {
		ExecutorService threadPool =  Executors.newSingleThreadExecutor();
		Future<String> future =
			threadPool.submit(
				new Callable<String>() {
					public String call() throws Exception {
						Thread.sleep(2000);
						return "hello";
					};
				}
		);
		System.out.println("等待线程执行");
		try {
			System.out.println("拿到结果:" + future.get());
		} catch (InterruptedException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (Exception e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
}

执行日志:

等待结果
拿到结果:hello
上述代码首先创建一个单线程池threadPool, threadPool提交submit一个Callable对象,这个Callbale对象里的call()方法相当于Runnable的Run()方法,是真正执行耗时任务的地方。
future.get()是一个阻塞等待 threadPool中的那个子线程执行完毕,并拿到结果:子线程返回的“hello”(Futrue<String>中的String表示返回一个字符串)。这里要区分的是future.get()阻塞主线程,但是线程池执行任务时,本身并不阻塞主线程的运行。

2. 等待线程池种的多个线程执行完毕

package testFuture;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Random;
import java.util.concurrent.Callable;
import java.util.concurrent.CompletionService;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorCompletionService;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;

public class futrue {

	/**
	 * @param args
	 */
	public static void main(String[] args) {
	
		SimpleDateFormat df = new SimpleDateFormat("yyyy-mm-dd hh:mm:ss");

		ExecutorService threadPool2 =  Executors.newFixedThreadPool(3);
		CompletionService<Integer> completionService = new ExecutorCompletionService<Integer>(threadPool2);
		System.out.println(
				df.format(new Date())+"开始添加3个任务");
		for(int i=1;i<=3;i++){
			final int seq = i;
			completionService.submit(new Callable<Integer>() {
				@Override
				public Integer call() throws Exception {
					Thread.sleep((5000));
					return seq;
				}
			});
		}
		for(int i=0;i<3;i++){
			try {
				System.out.println(
						completionService.take().get());
				System.out.println(
					df.format(new Date()));
			} catch (InterruptedException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			} catch (ExecutionException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}
		}
		System.out.println("over");
	}

	

}

执行日志:

2021-24-17 07:24:21开始添加3个任务
2
2021-24-17 07:24:26
1
2021-24-17 07:24:26
3
2021-24-17 07:24:26
over
 

上述我们创建了一个线程池,池子里有3个线程,接着我们用线程池创建一个CompletionService对象,向CompletionService里submit了3个任务,实质是为线程池添加了3个要执行的任务,每个任务耗时5秒。CompletionService.take()方法返回的是一个Futrue对象,同样Futrue.get()方法会阻塞等待其中一个线程执行完它所负责的任务,最终打印出了3,2,1的顺序,就是说任务3第一个被完成,接着Futrue.get()等待第2个任务完成,最后是第1个任务被完成。顺序不一样是由于多线程执行顺序的不确定性(CPU分时执行)。

等待所有任务执行完毕后,打印出了over.

一个常用应用场景是:当你用  “多线程池”  上传  “多张图片”  到服务器时,如果需要监听所有图片已被上传完毕,则可以使用这个Futrue。

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

冉航--小虾米

希望得到您的鼓励和交流

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值