6.实现 Callable 接口

前言

本篇章来介绍一下创建线程的第三种方式,其中创建线程一共有四种方式:

  • 继承 Thread 类
  • 实现 Runnable 接口
  • 实现 Callable 接口
  • 使用线程池的方式

那么下面我们来介绍一下 实现 Callable 接口的方式。

Callable 接口

- Java 5.0  java.util.concurrent 提供了一个新的创建执行线程的方式:Callable 接口

- Callable 接口类似于 Runnable,两者都是为那些其实例可能被另一个线程执行的类设计的。但是 Runnable 不会返回结果,并且无法抛出经过检查的异常。

- Callable 需要依赖FutureTask ,FutureTask 也可以用作闭锁。
  • 与使用​​Runnable​​相比, ​​Callable​​功能更强大些
  • 相比​​run()​​方法,可以有返回值
  • 方法可以抛出异常
  • 支持泛型的返回值
  • 需要借助​​FutureTask​​类,比如获取返回结果
  • ​Future​​接口
  • 可以对具体​​Runnable​​、​​Callable​​任务的执行结果进行取消、查询是否完成、获取结果等。
  • ​FutrueTask​​​是​​Futrue​​接口的唯一的实现类
  • ​FutureTask​​​ 同时实现了​​Runnable​​, ​​Future​​接口。它既可以作为​​Runnable​​被线程执行,又可以作为​​Future​​得到​​Callable​​的返回值

代码示例

1.创建一个实现Callable的实现类, 可以通过设置泛型,指定call方法返回的类型

import java.util.concurrent.Callable;

//1.创建一个实现Callable的实现类, 可以通过设置泛型,指定call方法返回的类型
class CallableThread implements Callable<Integer> {

@Override
public Integer call() throws Exception {
return null;
}

}

2.实现call方法,将此线程需要执行的操作声明在call()中

//1.创建一个实现Callable的实现类, 可以通过设置泛型,指定call方法返回的类型
class CallableThread implements Callable<Integer> {

//2.实现call方法,将此线程需要执行的操作声明在call()中
@Override
public Integer call() throws Exception {
int sum = 0;
for (int i = 0; i < 100; i++) {
if (i % 2 == 0){
sum += i;
}
}
return sum;
}

}

3.创建Callable接口实现类的对象

public class TestCallable {

public static void main(String[] args) {
//3.创建Callable接口实现类的对象
CallableThread callableThread = new CallableThread();
}

}

4.将此Callable接口实现类的对象作为传递到FutureTask构造器中,创建FutureTask的对象

import java.util.concurrent.Callable;
import java.util.concurrent.FutureTask;

public class TestCallable {
public static void main(String[] args) {
//3.创建Callable接口实现类的对象
CallableThread callableThread = new CallableThread();
//4.将此Callable接口实现类的对象作为传递到FutureTask构造器中,创建FutureTask的对象
FutureTask<Integer> futureTask = new FutureTask<>(callableThread);
}
}

5.将FutureTask的对象作为参数传递到Thread类的构造器中,创建Thread对象,并调用start()

public class TestCallable {
public static void main(String[] args) {
//3.创建Callable接口实现类的对象
CallableThread callableThread = new CallableThread();
//4.将此Callable接口实现类的对象作为传递到FutureTask构造器中,创建FutureTask的对象
FutureTask<Integer> futureTask = new FutureTask<>(callableThread);
//5.将FutureTask的对象作为参数传递到Thread类的构造器中,创建Thread对象,并调用start()
new Thread(futureTask).start();
}
}

6.获取Callable中call方法的返回值(因为会等待线程结束后再获取,所以可以当作闭关锁使用)

public class TestCallable {
public static void main(String[] args) {
//3.创建Callable接口实现类的对象
CallableThread callableThread = new CallableThread();
//4.将此Callable接口实现类的对象作为传递到FutureTask构造器中,创建FutureTask的对象
FutureTask<Integer> futureTask = new FutureTask<>(callableThread);
//5.将FutureTask的对象作为参数传递到Thread类的构造器中,创建Thread对象,并调用start()
new Thread(futureTask).start();
//6.获取Callable中call方法的返回值(因为会等待线程结束后再获取,所以可以当作闭关锁使用)
//get()返回值即为FutureTask构造器参数Callable实现类重写的call()的返回值。
try {
Integer sum = futureTask.get();
System.out.println("计算 sum = " + sum);
} catch (Exception e) {
e.printStackTrace();
}
}
}

执行效果如下:


6.实现 Callable 接口_多线程

image-20201102083840540