Java中的Callable实现多线程详解
作者:Orange#
Java中的Callable实现多线程
从Java提供多线程开始,最初的方案就是依靠Runnable接口定义线程类核心功能,但是Runnable中的run方法有一个缺点:该方法没有返回值。
从JDK1.5开始,在JUC(java.util.concurrent)包中提供了一个新的多线程实现接口:
@FunctionalInterface public interface Callable<V>{ public V call() throws Exception; }
接口Callable中有一个call方法,其返回值类型为V,这是一个泛型。
值得关注的是这个call方法有返回值,这意味着线程执行完毕后可以将处理结果返回。
根据以往的套路,我们要创建一个实现Callable的的类,将其作为线程主体,这个类要覆写call方法(其实call方法和run方法类似):
class MyThread4 implements Callable<String>{ // 定义核心业务类 private String name; public MyThread4(String name) { this.name = name; } @Override public String call() throws Exception { for(int i = 0 ; i < 100; i++) { System.out.println(this.name + "线程运行, x = " + i); } return "执行完毕"; // 返回值 } }
定义好了相应的线程执行类,就要用Thread的start()方法启动线程,但是表面上看Callable和Thread并没有联系,这需要我们深入的挖掘源码:
1、我们先从Java API中的java.util.concurrent包内找到Future<V>
public interface Future<V>{ V get() throws InterruptedException, ExecutionException; }
可以发现接口Future中有一个get()方法,注意:这个get()方法可以获取callable方法的返回值。
那现在我们可以获取Callable的返回值,但是,依旧看不出和Thread的关系。
2、再回到java.util.concurrent包中,找到RunnableFuture<V>
public interface RunnableFuture<V> extends Runnable, Future<V>{ void run(); }
这里我们先不关注run()方法,注意接口RunnableFuture继承了接口Runnable和接口Future<V>。
3、查看Runnable的实现类,发现了FutureTask<V>,FutureTask<V>有两个构造方法,我们关注其中一个:
public FutureTask(Callable<V> callable)
这个构造方法可传callable的实现类,而FutureTask又继承了Runnable。
所以Callable传给FutureTask,FutureTask又继承了Runnable,FutureTask就能作为Runnable传给Thread,start()方法就得以调用。
Callable与Runnable两个多线程接口最大的差别在于Callable有返回值。
实际使用哪个要看具体的需求。
完整代码:
class MyThread4 implements Callable<String>{ // 定义核心业务类 private String name; public MyThread4(String name) { this.name = name; } @Override public String call() throws Exception { for(int i = 0 ; i < 100; i++) { System.out.println(this.name + "线程运行, x = " + i); } return "执行完毕"; // 返回值 } } public class CallableTest { public static void main(String[] args) throws Exception{ Callable<String> callA = new MyThread4("线程A"); Callable<String> callB = new MyThread4("线程B"); Callable<String> callC = new MyThread4("线程C"); FutureTask<String> futureA = new FutureTask<String>(callA); FutureTask<String> futureB = new FutureTask<String>(callB); FutureTask<String> futureC = new FutureTask<String>(callC); new Thread(futureA).start(); new Thread(futureB).start(); new Thread(futureC).start(); System.out.println("A执行返回结果:" + futureA.get()); System.out.println("B执行返回结果:" + futureB.get()); System.out.println("C执行返回结果:" + futureC.get()); } }
到此这篇关于Java中的Callable实现多线程详解的文章就介绍到这了,更多相关Java的Callable多线程内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!
您可能感兴趣的文章:
- Java使用多线程批次查询大量数据(Callable返回数据)方式
- Java通过Callable实现多线程
- Java多线程中Callable和Future的解读
- Java使用Callable接口实现多线程的实例代码
- Java多线程实现之Callable详解
- Java中Runnable和Callable分别什么时候使用
- Java中Runnable与Callable接口的区别详解
- 详解Java中Callable和Future的区别
- Java使用Runnable和Callable实现多线程的区别详解
- java面试常问的Runnable和Callable的区别
- Java并发教程之Callable和Future接口详解
- Java中callable的实现原理