Java多线程带返回值的Callable接口java
在面试的时候,有时候是否是会遇到面试会问你,Java中实现多线程的方式有几种?你知道吗?你知道Java中有能够返回值的线程吗?在具体的用法你知道吗?若是两个线程同时来调用同一个计算对象,计算对象的call方法会被调用几回你知道吗?若是这些你知道,那么凯哥(凯哥Java:kaigejava)恭喜你,本文你能够不用看了。若是你不知道这些,那么凯哥一样要恭喜你,看了凯哥这篇文章以后,就知道这些了。来看看这篇文章咱们能学到什么面试
本节主要内容设计模式
一:三种获取多线程的的写法多线程
二:分析第三种写法的思想思路-使用了适配器模式ide
三:第三种方法怎么使用spa
四:多个线程调用同一个futrueTask后,future的call方法会被执行几回?线程
咱们已经知道Java中经常使用的两种线程实现方式:分别是继承Thread类和实现Runnable接口。设计
以下图:3d
从上图中,咱们能够看到,第三种实现Callable接口的线程,并且还带有返回值的。咱们来对比下实现Runnable和实现Callable接口的两种方式不一样点:orm
1:须要实现的方法名称不同:一个run方法,一个call方法
2:返回值不一样:一个void无返回值,一个带有返回值的。其中返回值的类型和泛型V是一致的。
3:异常:一个无需抛出异常,一个须要抛出异常。在后面使用场景中,凯哥会讲解到的
咱们先来看看Thread类:这个类是Java中获取线的对象。通常咱们获取并启动线程调用的是start方。从JDK的API中,咱们能够看到,start方法是JVM调用的
再来看看常写的方法:
Thread t1 = new Thread();
t1.start();
咱们来看看其构造器:
三个构造器:无参构造器、一个参数构造器和两个参数构造器。可是就没有咱们Callable做为参数的构造器。那么,咱们想要获取到线程,经过callable怎么获取呢 ?
就拿凯哥刚到帝都找房子的案例来讲吧。凯哥刚到帝都人生地不熟的,想要找房子怎么办呢?
房东有房子,凯哥想要找房子,那么这两个原本没有直接联系的经过房屋中介公司就产生了关系。凯哥要想找到房子,先要找到房屋中介公司,而后房屋中介公司又有房东的联系方式,而后凯哥就经过中介公司租到房东的房子了(中介公司从中间收取手续费)。这个现实案例我想你们都遇到过吧。
好了,咱们经过上面案例在回到Thread类和Callable类来看,这两个对象之间有没有中间商呢?
从上图中咱们发现,Threa的有参构造都是Runnable接口的。那么,有没有一个类既实现了Runnable接口又实现了Callable接口呢?若是有这样的一个类存在的话,callable就与Thread类产生了关系,就可使用了。咱们来看看Runnable接口的API吧
咱们能够已知的子类有个RunnableFuture<V>。这个接口的形式和咱们Callable接口的形式很像啊,以下图:
咱们从上图对比中能够看到,两个接口中的V都是方法返回值的类型。那么Callable和Thread两个类之间的桥梁就是这个类(RunnableFuture)或者是这个类的子类呢?咱们接着来看看这个对象的子类。
其中SwingWorker这个咱们不用看。这个是图形化的Swing相关的。咱们不用,那么咱们就来看看FutureTask这个类:
从这个类中,咱们能够看到其实现了Runnable接口,在构造器中,咱们能够看到:
FutureTask(Callable<V> callable)
建立一个 FutureTask ,它将在运行时执行给定的 Callable 。
以下图:
这个类是否是既有Callable接口又有Runnable接口了?这个就是咱们的中间类。
因此,咱们经过上面分析就能够获得下图的关系:
这种就是设计模式中的适配器模式(PS:在后面,凯哥会从新分享23种设计模式的)。在Java中的中间商是不会赚取差价的,放心。O(∩_∩)O
知道了Callable的设计思路以后,那么咱们怎么来使用呢?
步骤:
1:一样建立一个类实现Callable接口;
2:经过futureTask类使用其传递Callable接口做为参数的有参构造方法;
3:使用thread的有参构造;
4:t1.start()启动线程
5:启动线程后,经过futureTask.get()方法获取到线程的返回值。
以下图:
咱们来查看运行结果:
进入了callable接口且获取到了返回值:1024.说明callable的使用正确了。
须要注意:futrueTask.get()方法放到最后,这样就不会影响主线程了。若是get方法放在前面的话,会形成主线程阻塞,等到futrueTask运行完成以后,才继续执行本身的逻辑。这样就失去了开启线程的意义了!!!
咱们能够看到t1和t2都start了,说明两个线程都启动了。并且都是用的是同一个futureTask对象。问题:MyThread3中的call方法会被调用几回呢?