Java使用Executor执行Callable任务时的几种方法

多线程在需要返回值时,我们知道需要用到Callable和Future。Callable的cell方法可以返回一个值并且可抛出异常,是对Runnable的很好的补充;Future表示了一个任务的周期,它提供了判断任务状态、获取任务结果和取消任务等方法 。
下面演示三种使用Executor执行Callable任务的方法。

     /**
     * 测试任务,返回任务的序号
     */
    public static class TestTask implements Callable<Integer>{
        int index;
        public TestTask(int index) {
            this.index = index;
        }
        @Override
        public Integer call() throws Exception {
            return index;
        }
    }
/**
     * 方法一:手动的保存任务的返回,这样的好处是每个任务对应的结果我们很清楚
     */
    @Test
    public void ordinaryTest() throws ExecutionException, InterruptedException {
        ExecutorService es = Executors.newFixedThreadPool(10);
        List<Future<Integer>> futures = new ArrayList<>();
        for(int i = 0; i < 10; i++) {
            TestTask testTask = new TestTask(i);
            Future<Integer> future = es.submit(testTask);
            futures.add(future);
        }
        es.shutdown();
        for(int i = 0; i < 10; i++) {
            System.out.println("index:" + i + ",future:"+ futures.get(i).get());
        }
    }
   /**
    * 方法二:使用ExecutorCompletionService
    * ExecutorCompletionService中使用阻塞队列保存各任务的返回结果,返回是无序的,即谁先执行完成(异常、中断),谁先入队。
    * 当我们不关心结果的顺序,或者需要一个任务完成时就取消其他任务的情况下,它是非常的方便的
    */
   @Test
   public void completionServiceTest() throws ExecutionException, InterruptedException {
       ExecutorService es = Executors.newFixedThreadPool(10);
       CompletionService<Integer> completionService = new ExecutorCompletionService<>(es);
       for(int i = 0; i < 10; i++) {
           TestTask testTask = new TestTask(i);
           completionService.submit(testTask);
       }
       es.shutdown();
       for(int i = 0; i < 10; i++) {
           Future<Integer> future = completionService.take();
           System.out.println("index:" + i + ",future:"+ future.get());
       }
   }
    /**
     * 方法三:ExecutorService的invokeAll方法
     * invokeAll方法入参为一组任务,返回一组Future,这两个集合是有相同结构的,
     * 即它是按照入参的任务集合中迭代器的顺序将所有的Future添加到返回的集合中,从而任务和Future在它们各自的集合中有着同样的顺序。
     * 当我们需要任务和结果的对应关系时,使用invokeAll方法来代替第一种方法
     */
    @Test
    public void invokeAllTest() throws InterruptedException, ExecutionException {
        ExecutorService es = Executors.newFixedThreadPool(10);
        List<TestTask> tasks = new ArrayList<>();
        for (int i = 0; i < 10; i++){
            tasks.add(new TestTask(i));
        }
        List<Future<Integer>> futures = es.invokeAll(tasks);
        es.shutdown();
        for (int i = 0; i < futures.size(); i++){
            System.out.println("index:" + i + ",future:"+ futures.get(i).get());
        }
    }

下面看一下ExecutorCompletionService的原理:
ExecutorCompletionService是将Executor和BlockingQueue的功能融合在一起,可将Callbale任务提交给它来执行,然后我们就可以像队列一样使用take或poll来得到已经完成的任务结果。下面是源码分析:

    /**
     *ExecutorCompletionService包含三个成员变量,最主要的是completionQueue,它的类型阻塞队列
     */
 
    private final Executor executor;
    private final AbstractExecutorService aes;
    private final BlockingQueue<Future<V>> completionQueue;
 
    /**
     * 构造方法,需要我们传入一个Executor
     */
    public ExecutorCompletionService(Executor executor) {
        if (executor == null)
            throw new NullPointerException();
        this.executor = executor;
        this.aes = (executor instanceof AbstractExecutorService) ?
                (AbstractExecutorService) executor : null;
        this.completionQueue = new LinkedBlockingQueue<Future<V>>();
    }
 
    /**
     * 提交任务的方法,其中的RunnableFuture为一个内部类,继承自FutureTask
     */
    public Future<V> submit(Callable<V> task) {
        if (task == null) throw new NullPointerException();
        RunnableFuture<V> f = newTaskFor(task);
        executor.execute(new QueueingFuture(f));
        return f;
    }
    private class QueueingFuture extends FutureTask<Void> {
        QueueingFuture(RunnableFuture<V> task) {
            super(task, null);
            this.task = task;
        }
 
        /**
         * 这是QueueingFuture存在的主要原因,当任务执行完成后,将任务结果装入队列中
         */
        protected void done() { completionQueue.add(task); }
        private final Future<V> task;
    }
 
    /**
     * 从队列中获取返回值
     */
    public Future<V> take() throws InterruptedException {
        return completionQueue.take();
    }

转载于:https://my.oschina.net/u/2424727/blog/2032393

### 回答1: 使用 Java 实现线程有两种方法: 1. 实现 Runnable 接口:通过实现 Runnable 接口并重写其 run() 方法来创建线程。 2. 继承 Thread 类:通过继承 Thread 类并重写其 run() 方法来创建线程。 两种方法都是可行的,但是通常推荐使用实现 Runnable 接口的方法,因为继承了 Thread 类会导致继承树不再继续向上延伸,进而影响代码的复用性。 ### 回答2: 在Java中,实现线程有三种主要方法: 1. 继承Thread类:这是最基本的方法,需要创建一个继承自Thread类的子类,并重写其中的run()方法。run()方法中定义了线程执行的具体逻辑。然后可以通过创建子类对象,并调用其start()方法来启动线程。 2. 实现Runnable接口:这是另一种常见的方法,需要创建一个实现了Runnable接口的类,并实现其中的run()方法。然后可以将该实现类的对象作为参数传递给Thread类的构造方法,在构造Thread对象将其包装成一个线程对象。然后调用线程对象的start()方法来启动线程。 3. 实现Callable接口:这是Java5版本之后引入的新特性。Callable接口与Runnable接口相似,都是定义了一个run()方法执行线程的逻辑。但是不同的是,run()方法的返回值为void,而Callable的call()方法可以有返回值。使用Callable需要创建一个实现了Callable接口的类,并实现其中的call()方法。然后可以通过将该实现类的对象作为参数传递给ExecutorService类的submit()方法来启动线程,并获取call()方法的返回值。 以上是最常见的三种方法来实现线程。不同的方法适用于不同的场景,通过选择合适的方法可以更灵活地管理和控制线程的执行。 ### 回答3: 在Java中,有三种主要的方法可以实现线程: 1. 继承Thread类:创建一个新的类,继承Thread类,并重写run()方法。在run()方法中定义线程的逻辑和动作。然后可以创建该类的实例,并调用start()方法来启动新的线程。 2. 实现Runnable接口:创建一个新的类,实现Runnable接口,并重写run()方法。在run()方法中定义线程的逻辑和动作。然后可以创建Thread对象,将实现了Runnable接口的类的实例作为参数传递给Thread的构造方法,最后调用start()方法来启动新的线程。 3. 使用Executor框架:Executor框架是Java提供的线程池管理工具。它利用ExecutorService接口来管理线程池。可以使用Executors类中的一些静态方法来创建不同类型的线程池。然后使用submit()方法提交一个实现了Runnable接口或Callable接口的对象给线程池,线程池会自动分配线程来执行任务。 无论使用哪种方法,实现线程的本质都是通过创建Runnable对象,然后通过线程启动器(Thread类的start()方法Executor框架)来调度执行。不过,继承Thread类更加直观,但不方便资源共享;而实现Runnable接口可以避免单继承局限,并且可以共享资源;使用Executor框架可以进一步简化线程管理和资源分配。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值