Java中的Callable实现多线程详解
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的实现原理
相关文章
谈谈Java中整数类型(short int long)的存储方式
在java中的整数类型有四种,分别是byte short in long,本文重点给大家介绍java中的整数类型(short int long),由于byte只是一个字节0或1,在此就不多说了,对java中的整数类型感兴趣的朋友一起学习吧2015-11-11解决springboot报错Could not resolve placeholder‘x
这篇文章主要介绍了解决springboot报错:Could not resolve placeholder ‘xxx‘ in value “${XXXX}问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教2023-11-11Java使用@Autowired注解获取对象为null的几种情况及解决方法
这篇文章主要给大家介绍了使用@Autowired注解获取对象为null的几种情况以及 解决方法,文中有详细的代码示例讲解,具有一定的参考价值,需要的朋友可以参考下2023-09-09mybatis-plus3.0.1枚举返回为null解决办法
这篇文章主要介绍了mybatis-plus3.0.1枚举返回为null解决办法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧2020-12-12idea中开启Run Dashboard 和 快速复制项目并改变端口的方法
这篇文章主要介绍了idea中开启Run Dashboard 和 快速复制项目并改变端口的方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下2020-08-08
最新评论