线程--Callable和Future

本文介绍了三种创建线程的方法:直接使用Thread类、实现Runnable接口及使用Callable接口配合Future获取返回值。并详细探讨了Future接口的功能及其实现类FutureTask的用法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

+ 创建线程:
- 通过Thread直接创建
- 通过Runnable接口创建 
- 通过Callable和Future接口创建。
* Callable是类似于Runnable的接口,实现Callable接口的类和实现Runnable的类都是可被其他线程执行的任务。 
+ Callable和Runnable的区别如下:  
- Callable定义的方法是call,而Runnable定义的方法是run。
- Callable的call方法可以有返回值,而Runnable的run方法不能有返回值。
- Callable的call方法可抛出异常,而Runnable的run方法不能抛出异常。 
- Callable接口有范型限制,Callable接口里的泛型形参类型与call方法的反回类型相同,而且Callable接口是函数式接口,因此可以用Lambda表达式创建Callable对象。
+ Future表示异步计算的结果,它提供了检查计算是否完成的方法,以等待计算的完成,并检索计算的结果。Future的cancel方法取消任务的执行,有一个布尔参数,参数为true表示立即中断任务的执行,参数为false表示允许正在运行的任务运行完成。Future的get方法等待计算完成,获取计算结果。

    - Future只是一个接口,所以是无法直接用来创建对象使用的。FutureTask是Future接口的一个实现类,它实现了RunnableFuture接口继承的Runnable接口和Future接口,因此既可以作为Runnable被线程执行,又可以作为Future得到Callable的返回值。


代码示例:

 

 

 

 

import java.util.List;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.FutureTask;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;


public class CallableAndFuture {


    /**
     * 自定义一个任务类(求前n项和),实现Callable接口
     */
    public static class SumNumbers implements Callable<Integer> {


        private int val = 0;


        public SumNumbers(int val){
            this.val = val;
        }


        public Integer call() throws Exception {
            int sum = 0;
            for (int i = 0; i <= val; ++i) {
                if (Integer.MAX_VALUE - sum < i) {
                    throw new RuntimeException("overflow: " + sum + "+" + i + ">" + Integer.MAX_VALUE);
                }
                sum += i;
                TimeUnit.MICROSECONDS.sleep(1);
            }
            return sum;
        }
    }


    public static void testInvokeAny() {
        List<SumNumbers> tasks = new ArrayList<SumNumbers>();
        tasks.add(new SumNumbers(100));
        tasks.add(new SumNumbers(200));
        tasks.add(new SumNumbers(Integer.MAX_VALUE));
        tasks.add(new SumNumbers(Integer.MAX_VALUE));
        tasks.add(new SumNumbers(5000));


        // 创建一个执行任务的服务
        ExecutorService es = Executors.newFixedThreadPool(tasks.size());
        System.out.println("invoke any tasks!!!");
        Integer result = 0;
        try {
            result = es.invokeAny(tasks, 1, TimeUnit.MILLISECONDS);
        } catch (InterruptedException e) {
            e.printStackTrace();
        } catch (ExecutionException e) {
            e.printStackTrace();
        } catch (TimeoutException e) {
            e.printStackTrace();
        }
        // Integer result = es.invokeAny(tasks);
        System.out.println(String.format("result = %d \n", result));
        // 停止任务执行服务
        es.shutdownNow();
    }


    public static void testInvokeAll() {
        List<SumNumbers> tasks = new ArrayList<SumNumbers>();
        tasks.add(new SumNumbers(100));
        tasks.add(new SumNumbers(200));
        tasks.add(new SumNumbers(Integer.MAX_VALUE));
        tasks.add(new SumNumbers(Integer.MAX_VALUE));
        tasks.add(new SumNumbers(5000));


        // 创建一个执行任务的服务
        ExecutorService es = Executors.newFixedThreadPool(tasks.size());
        try {
            System.out.println("invoke any tasks!!!");
            // Integer result = es.invokeAny(tasks, 5, TimeUnit.SECONDS);
            Integer result = es.invokeAny(tasks);
            System.out.println(String.format("result = %d \n", result));


            TimeUnit.SECONDS.sleep(1);
            System.out.println("invoke all tasks!!!");
            List<Future<Integer>> results = es.invokeAll(tasks, 1000, TimeUnit.MILLISECONDS);
            // TimeUnit.SECONDS.sleep(10);
            System.out.println("Begin to get results...");
            long t1 = System.currentTimeMillis();
            for (int i = 0; i < results.size(); ++i) {
                // 线程可能在submit和get之间的运行中出现异常,但是Future<T>会记录结果和异常
                // 获取结果方法1: 防止task运行总出现异常了呢 这里要处理一下
                try {
                    System.out.println("result " + i + ", isDone=" + results.get(i).isDone() + ", isCanceled= "
                                       + results.get(i).isCancelled()); // 一般都是true
                    System.out.println("result " + i + " : " + results.get(i).get());
                    TimeUnit.MILLISECONDS.sleep(10);
                } catch (Throwable e) {
                    System.err.println("skipp failed tasks : \n" + e.toString() + "\n\n");
                }
                // 获取结果方法2:通过isCancelled来判断是否正常结束
                if (!results.get(i).isCancelled()) {
                    System.out.println("Result completed normally  " + i + " : " + results.get(i).get());
                }
            }
            long t2 = System.currentTimeMillis();
            System.out.println("Time-consuming=" + (t2 - t1) + " ms");
        } catch (Exception e) {
            System.out.println("error");
            System.out.println(e.toString());
        }
        // 停止任务执行服务
        es.shutdownNow();
    }


    /**
     * FutureTask封装了结果
     */
    public static void testFutureTask() {
        try {
            FutureTask<Integer> task = new FutureTask<Integer>(new SumNumbers(100));
            // 创建一个执行任务的服务
            ExecutorService es = Executors.newFixedThreadPool(1);
            es.submit(task);


            // 正确的获取结果
            System.out.println(task.get());


            // 下面后去result的方法是不对的
            Future<?> result = es.submit(task);
            if (result.get() == null) {
                System.out.println("the result is in the task object!");
            }
            es.shutdown(); // must shutdown ExecutorService
        } catch (Exception e) {
            e.printStackTrace();
        }


    }


    public static void main(String[] args) {
        // testFutureTask();
        testInvokeAll();
        testInvokeAny();
    }
}

 









注意:
ExecutorService.submit或者invokeAll的Task,如果超时没有返回结果的线程,那么ExecutorService会cancel这些线程(stop 这些线程),不会让这些线程继续运行下去。

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值