ThreadPoolExecutor之Future、FutureTask源码解析

1、Future的出现原因

我们在Java线程池ThreadPoolExecutor详解中利用execute(Runnable r)方法来异步执行任务,但是有一个缺点,就是无法执行带有返回值的任务。

所以引入了Future来解决这个问题。

2、Future详解

Future中有5个方法:
在这里插入图片描述

1、cancel():取消任务
2、isCancelled():任务是否取消成功
3、isDone():任务是否完成了
4、get():获取返回值,阻塞
5、get(long count,TimeUnit):按照count个unit超时时间来获取返回值,阻塞

ThreadPoolExecutor提供了三个方法,来获取返回值
在这里插入图片描述
1、submit(Runnable r)
因为Runnable的run方法是void的,没有返回值,所以Future.get()是null。

源码

static final class RunnableAdapter<T> implements Callable<T> {
    final Runnable task;
    final T result;
    RunnableAdapter(Runnable task, T result) {
        this.task = task;
        this.result = result;
    }
    public T call() {
        task.run();
        return result;
    }
}

将runnable封装为一个自定义的Callable对象,result为null,具体的流程可以看最底下的源码。

2、submit(Runnable r,T result)
T result就是传入的对象。我们可以通过自定义Runnable来将result传入到自定义的Runnable中,在run()方法中对结果写入到result中。其中返回的对象future.get()==result。

例子如下:

public class TestThreadExecutor {

    public static void main(String[] args) throws ExecutionException, InterruptedException {

        BlockingQueue queue = new LinkedBlockingQueue(10);
        ThreadPoolExecutor executor = new ThreadPoolExecutor(4,10,2,TimeUnit.SECONDS,queue);

        //设置coreThread如果超时也会被取消
        executor.allowCoreThreadTimeOut(true);

        int[] arr = new int[2];
        MyRunnable runnable = new MyRunnable(arr);
        Future f = executor.submit(runnable,arr);
        System.out.println(Arrays.toString((int[])f.get()));
    }
}

class MyRunnable implements Runnable {

    private int[] arr;

    public MyRunnable(int[] arr){
        this.arr = arr;
    }

    @Override
    public void run() {
        arr[0] = 111;
    }
}

在这里插入图片描述
源码

static final class RunnableAdapter<T> implements Callable<T> {
    final Runnable task;
    final T result;
    RunnableAdapter(Runnable task, T result) {
        this.task = task;
        this.result = result;
    }
    public T call() {
        task.run();
        return result;
    }
}

将runnable封装为一个自定义的Callable对象,result为传入的对象,具体的流程可以看最底下的源码。

3、submit(Callable c)
submit(Runnable r,T result)不太好用,需要每次都自定义Runnable和result,太麻烦了。
Callable更加好用,传入一个Callable对象就行。

Callable和Runnable是大致一样,只不过Callable能够返回值。

Callable callable = new Callable() {
    @Override
    public Object call() throws Exception {
        String name = "777";
        name += "6666";
        return name;
    }
};

Future f = executor.submit(callable);

System.out.println(f.get());

在这里插入图片描述
从上述代码可以看出,Callable中call()返回的对象就是future.get()得到的对象。

源码解析

public <T> Future<T> submit(Callable<T> task) {
    if (task == null) throw new NullPointerException();
    RunnableFuture<T> ftask = newTaskFor(task);//利用callable新建了一个FutureTask
    execute(ftask);//会调用ftask.run()方法
    return ftask;//返回一个FutureTask对象
}

protected <T> RunnableFuture<T> newTaskFor(Callable<T> callable) {
    return new FutureTask<T>(callable);
}

public FutureTask(Callable<V> callable) {
    if (callable == null)
        throw new NullPointerException();
    this.callable = callable;
    this.state = NEW;  
}

public void run() {
    if (state != NEW ||
        !UNSAFE.compareAndSwapObject(this, runnerOffset,
                                     null, Thread.currentThread()))
        return;
    try {
        Callable<V> c = callable;
        if (c != null && state == NEW) {
            V result;
            boolean ran;
            try {
                //执行callable.call()
                result = c.call();
                ran = true;
            } catch (Throwable ex) {
                result = null;
                ran = false;
                setException(ex);
            }
            if (ran)
                //将outcome设置为result
                set(result);
        }
    } finally {
        runner = null;
        int s = state;
        if (s >= INTERRUPTING)
            handlePossibleCancellationInterrupt(s);
    }
}

protected void set(V v) {
    if (UNSAFE.compareAndSwapInt(this, stateOffset, NEW, COMPLETING)) {
        //这里这里这里
        outcome = v;
        UNSAFE.putOrderedInt(this, stateOffset, NORMAL); // final state
        finishCompletion();
    }
}

future.get()

public V get() throws InterruptedException, ExecutionException {
    int s = state;
    if (s <= COMPLETING)
        //阻塞
        s = awaitDone(false, 0L);
    //这里这里这里
    return report(s);
}

private V report(int s) throws ExecutionException {
    //这里这里这里
    Object x = outcome;
    if (s == NORMAL)
        //这里这里这里
        return (V)x;
    if (s >= CANCELLED)
        throw new CancellationException();
    throw new ExecutionException((Throwable)x);
}

综上所述,还是submit(Callable c)比较好使!

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
ThreadPoolExecutor是一个用于管理线程池的类。它的构造函数和execute方法是源码中的两个重要部分。构造函数用于初始化线程池,而execute方法用于提交任务给线程池进行执行。\[1\] 在构造函数中,我们可以看到线程池的继承关系以及一些参数的设置。另外,Executors类提供了一些线程池的模板供我们使用,而拒绝策略的实现和任务队列BlockingQueue的实现也是构造函数中的重要内容。\[2\] execute方法是用来执行任务的,它的流程包括了一些重要的步骤。首先,它会判断线程池中的线程数量是否小于核心线程数,如果是,则会创建新的线程来执行任务。如果线程池中的线程数量已经达到了核心线程数,那么任务会被添加到任务队列中等待执行。如果线程池中的线程数量小于最大线程数,任务会被创建新的线程来执行。如果线程池中的线程数量已经达到了最大线程数,那么会根据拒绝策略来处理任务。\[3\] 在执行任务的过程中,会调用Worker的run方法来执行任务。Worker的run方法中包含了一些重要的逻辑,比如执行任务前的处理和执行任务后的处理。同时,getTask方法用于获取待执行的任务,其中也涉及到了线程最大空闲时间的设置。最后,拒绝策略会调用handler的rejectedExecution方法来处理被拒绝的任务。\[3\] 通过对ThreadPoolExecutor源码解析,我们可以更好地理解线程池的工作原理和内部机制。 #### 引用[.reference_title] - *1* *2* *3* [线程池(ThreadPoolExecutor源码解析](https://blog.csdn.net/li1325169021/article/details/121893030)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值