1.Callable接口简介
2.FutureTask类简介
3.Callable接口使用
1.Callable接口简介
在介绍Callable接口之前,我们先回顾一下我们很熟悉的Runnable接口:
我们想要使用线程的时候,很多时候都是先实现Runnable接口,然后重写run方法,如下所示:
public class RunnableDemo implements Runnable {
@Override
public void run() {
}
}
不难发现,我们一直使用的Runnable接口的run方法,是没有返回值的,如果我们现在有一个场景要使用多线程,并且需要有返回值怎么办?
此时,我们就引出有返回值的线程接口,Callable!
Callable: 返回线程执行的结果并且可能抛出异常的线程类!
这么说可能比较抽象,我们使用代码示例,就可以一下子就看明白了!
public class CallDemo implements Callable {//泛型类
@Override
public Integer call() throws Exception { //泛型类为返回值
return null;
}
}
现在我们看到这个代码示例就应该瞬间明白了,传入泛型类,那么call方法的返回值也会和泛型类。
好了,现在我们拥有callable接口了,但是如何使用呢?难道和Runnable接口一样,也是传入Thread()类吗?
我们从jdk8翻阅一下api字典 java api,发现并没有callable接口的传入参数,难道这是java的失误吗?
我们使用适配器模式的思想,如果有个类,同时可以将Runnable接口和Callable接口进行整合,是不是就满足我们这个需求了呢?
没错,是有这种类,它就是FutureTask类!能够提供返回值的异步计算!
2.FutureTask类简介
FutureTask,一个可获得返回值的异步计算类!可以调用方法去开始和取消一个计算,可以查询计算是否完成并且获取计算结果,只有当计算完成时才能获取到计算结果,否则就会阻塞!
3.Callable接口使用
可能看了上述讲解,我们还是不知道如何使用,接下来我们直接用一个Demo来进行示例:
我们先设定一个类,经过三秒中之后返回一个数值,代表计算过程。
public class CallableDemo implements Callable {
//和runnable的接口比起来,这是有返回值的接口
//还可以抛异常
@Override
public Integer call() throws Exception {
System.out.println(Thread.currentThread().getName() + "****进入callable了!");
Thread.sleep(3000);
return 1024;
}
}
接下来我们将用FutureTask获取这个运算结果:
//把实现callable的接口当做参数传入futuretask
FutureTask futureTask = new FutureTask<>(new CallableDemo());
//因为futureTask实现了Runnable接口,像普通的Runnable实现类一样传入Thread就可以了
Thread t1 = new Thread(futureTask, "t1");
//正常启动
t1.start();
//尝试获取返回结果
System.out.println("****** result=" + futureTask.get());
根据注释,Future可以很容易地就被使用了。
但是如果我们修改一下代码呢?
//把实现callable的接口当做参数传入futuretask
FutureTask futureTask = new FutureTask<>(new CallableDemo());
//因为futureTask实现了Runnable接口,像普通的Runnable实现类一样传入Thread就可以了
Thread t1 = new Thread(futureTask, "t1");
//让两个相同的线程同时执行futureTask
Thread t2 = new Thread(futureTask, "t2");
//正常启动
t1.start();
t2.start();
//尝试获取返回结果
System.out.println("****** result=" + futureTask.get());
运行结果为:
看,还是和刚才一样,也就是说一个FutureTask只能绑定一个Thread!绑定多了也是无效的!
总结:
Callable是一种可以拥有返回值的线程类。
优点:
可以获得任务执行返回值;
通过与Future的结合,可以实现利用Future来跟踪异步计算的结果。