使用 Callable 和 Future 创建线程

简介: 使用 Callable 和 Future 创建线程

任务描述

本关任务:通过 CallableFuture 来创建线程。

相关知识

Java1.5版本开始,就提供了 CallableFuture 来创建线程,这种方式也是在Java程序员面试中经常会被问到的问题。

上一小节介绍了ThreadRunnable两种方式创建线程,不过这两种方式创建线程都有一个缺陷:在执行完任务之后无法获取执行结果。 如果需要获取执行结果,就必须通过共享变量或者使用线程通信的方式来达到效果,这样使用起来就比较麻烦。

而如果使用CallableFuture,通过它们就可以在任务执行完毕之后得到任务执行结果

本小节你需要掌握的知识有:

1.什么是CallableFuture

2.如何通过CallableFuture创建线程。

Callable和Future

它们俩其实挺有意思,在运行的时候各司其职,Callable产生结果Future获取结果

使用步骤如下:

创建 Callable 接口的实现类,并实现 call() 方法,该 call() 方法将作为线程执行体,并且有返回值;
创建 Callable 实现类的实例,使用 FutureTask 类来包装 Callable 对象,该 FutureTask 对象封装了该 Callable 对象的 call() 方法的返回值;
使用 FutureTask 对象作为 Thread 对象的 target 创建并启动新线程;
调用 FutureTask 对象的 get() 方法来获得子线程执行结束后的返回值。

接下来通过一个示例来学习这两个对象的使用:

publicclassTest {
publicstaticvoidmain(String[] args) {
CallableThreadTestcts=newCallableThreadTest();
// 接收FutureTask<Integer>ft=newFutureTask<>(cts);
newThread(ft, "有返回值的线程").start();
for (inti=0; i<30; i++) {
System.out.println( "main"+" 的循环变量i的值:"+i);
        }
try {
System.out.println("子线程的返回值:"+ft.get());
        } catch (Exceptione) {
e.printStackTrace();
        }
    }
}
classCallableThreadTestimplementsCallable<Integer> {
publicIntegercall() throwsException {
inti=0;
for (; i<30; i++) {
System.out.println(Thread.currentThread().getName() +" "+i);
        }
returni;
    }
}

运行这段程序你应该可以获取到类似如下结果(每次运行的结果不一致): ... ... main 的循环变量i的值:28 main 的循环变量i的值:29 有返回值的线程 23 有返回值的线程 24 有返回值的线程 25 有返回值的线程 26 有返回值的线程 27 有返回值的线程 28 有返回值的线程 29 子线程的返回值:30

由于输出过长,省略了部分结果,可以发现在最后接收到了子线程的返回值。


在实现Callable接口中,此时不再是run()方法了,而是call()方法,此call()方法作为线程执行体,同时还具有返回值!

细心的你会发现这个结果是call函数的返回值,怎么拿到这个返回值的呢?是通过FutureTask拿到的,使用ft.get()方法即可获得线程的返回值,这就是一个简单的使用Callable和Future的过程了。

关于Callable和Future的使用,以及他们的常用函数,我们将会在后续的实训中学习。

编程要求

请仔细阅读右侧代码,根据方法内的提示,在Begin - End区域内进行代码补充,具体任务如下:

  • runThread(int num)函数中执行线程,创建Callable线程,Callable线程需要执行求第num项斐波那契数列的值,最后在runThread函数中获取Callable线程执行的结果,并打印输出。

斐波那契数列(Fibonacci数列)  这个数列从第3项开始,之后的每一项都等于它的前两项数字之和。 这个数列为: 1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89, 144, 233,377,610,987,1597,2584,4181,6765,10946,17711,28657,46368........

测试说明

补充完代码后,点击测评,平台会对你编写的代码进行测试,当你的结果与预期输出一致时,即为通过。

输入:3 输出:线程的返回值为:2

输入:5 输出:线程的返回值为:5


开始你的任务吧,祝你成功!


实现代码

注:可以在 ThreadCallable类中添加无参和有参构造方法

Task.java:

packagestep2;
importjava.util.concurrent.Callable;
importjava.util.concurrent.FutureTask;
publicclassTask {
publicvoidrunThread(intnum) {
// 请在此添加实现代码/********** Begin **********/// 在这里开启线程 获取线程执行的结果try {
ThreadCallabletc=newThreadCallable(num);
FutureTask<Integer>ft=newFutureTask<>(tc);
newThread(ft).start();
System.out.println("线程的返回值为:"+ft.get());
        } catch (Exceptione) {
// TODO 自动生成的 catch 块e.printStackTrace();
        }
/********** End **********/    }
}
//请在此添加实现代码/********** Begin **********//* 在这里实现Callable接口及方法 */classThreadCallableimplementsCallable {
intnum;
publicThreadCallable() {
super();
// TODO 自动生成的构造函数存根    }
publicThreadCallable(intnum) {
super();
this.num=num;
    }
@OverridepublicIntegercall() throwsException {
returnfun(num);
    }
publicintfun(intnum) {
if (num<3) {
return1;
        } elsereturnfun(num-1) +fun(num-2);
    }
}
/********** End **********/

Test.java:

packagestep2;
importjava.util.Scanner;
importjava.util.concurrent.Callable;
importjava.util.concurrent.FutureTask;
importstep2.ThreadCallable;
publicclassTest {
publicstaticvoidmain(String[] args) {
Scannersc=newScanner(System.in);
intnum=sc.nextInt();
ThreadCallablecallable=newThreadCallable();
if(!(callableinstanceofCallable)){
System.out.println("未定义Callable线程,或者定义错误");
        }
Tasktask=newTask();
task.runThread(num);
    }
}
目录
相关文章
|
4月前
|
存储 Java
高并发编程之多线程锁和Callable&Future 接口
高并发编程之多线程锁和Callable&Future 接口
66 1
|
23天前
|
并行计算 Java 大数据
Callable和Future
Callable和Future
|
4月前
|
安全 Java
多线程(CAS, ABA问题, Runnable & Callable & 僵尸线程 & 孤儿进程)
多线程(CAS, ABA问题, Runnable & Callable & 僵尸线程 & 孤儿进程)
42 1
|
4月前
面试官:除了继承Thread类和实现Runnable接口,你知道使用Callable接口的方式来创建线程吗?
面试官:除了继承Thread类和实现Runnable接口,你知道使用Callable接口的方式来创建线程吗?
35 0
面试官:除了继承Thread类和实现Runnable接口,你知道使用Callable接口的方式来创建线程吗?
|
4月前
|
Java 程序员
Java多线程基础-16:简述Java并发编程JUC中的Callable接口
Callable接口是Java中用于描述带有返回值任务的接口,与Runnable相对,后者无返回值。Callable的call()方法用于执行具体任务并返回结果。
65 0
|
4月前
|
Java
Java并发编程:理解并使用Future和Callable接口
【2月更文挑战第25天】 在Java中,多线程编程是一个重要的概念,它允许我们同时执行多个任务。然而,有时候我们需要等待一个或多个线程完成,然后才能继续执行其他任务。这就需要使用到Future和Callable接口。本文将深入探讨这两个接口的用法,以及它们如何帮助我们更好地管理多线程。
|
4月前
|
Java
创建线程的三种方式:继承Thread、Runnable 接口、Callable 接口
创建线程的三种方式:继承Thread、Runnable 接口、Callable 接口
|
4月前
|
Java
多线程创建方式三 - 实现Callable接口
多线程创建方式三 - 实现Callable接口
39 1
|
4月前
|
安全 Java C++
多线程之Callable接口、ReentrantLock、信号量 Semaphore以及CountDownLatch
多线程之Callable接口、ReentrantLock、信号量 Semaphore以及CountDownLatch
|
10月前
|
Java
ExecutorService、Callable、Future实现有返回结果的多线程原理解析
ExecutorService、Callable、Future实现有返回结果的多线程原理解析
63 0