异步模式模式Future(结合Callable可以获取线程返回结果)
submit 和 excute是有啥区别
如果有这样的需求:
多线程实现下载,提高效率。
不论是Thread类还是Runnable接口重写run方法,有个特点就是没有返回值~~~~~~
我都主线程 如何知道子线程程序执行完毕呢? 子线程执行完毕 立马告诉主线程 可以wait 和 notify
在Java中,创建线程一般有两种方式,一种是继承Thread类,一种是实现Runnable接口。然而,这两种方式的缺点是在线程任务执行结束后,无法获取执行结果。我们一般只能采用共享变量或共享存储区以及线程通信的方式实现获得任务结果的目的。
不过,Java中,也提供了使用Callable和Future来实现获取任务结果的操作。Callable用来执行任务,产生结果,而Future用来获得结果。
可以看到,与Runnable接口不同之处在于,call方法带有泛型返回值V。
submit返回 Future类型
package com.toov5.thread; import java.util.concurrent.Callable; import java.util.concurrent.ExecutionException; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.Future; public class CallableThreadTest { public static void main(String[] args) throws InterruptedException,ExecutionException { ExecutorService executorService=Executors.newCachedThreadPool(); //提交时候 线程池开始执行 System.out.println("1主线程开始执行"); Future<String> sFuture = executorService.submit(new TaskCallable()); System.out.println("hhhhh"); String ResultStirng = sFuture.get(); //下面线程执行的结果 System.out.println("2结果:"+ResultStirng); System.out.println("okokokok"); } } class TaskCallable implements Callable<String>{ //可以返回线程结果 @Override public String call() throws Exception { System.out.println("3正在执行任务,请等待五秒"); Thread.sleep(5000); System.out.println("4执行完毕"); return "这是返回结果"; } }
1和2是不需要阻塞的 后面的需要阻塞哈
子线程去执行 主线程爱干满干嘛不用专门去等待 子线程运行完毕 主获取结果
Future常用方法
V get() :获取异步执行的结果,如果没有结果可用,此方法会阻塞直到异步计算完成。
V get(Long timeout , TimeUnit unit) :获取异步执行结果,如果没有结果可用,此方法会阻塞,但是会有时间限制,如果阻塞时间超过设定的timeout时间,该方法将抛出异常。
boolean isDone() :如果任务执行结束,无论是正常结束或是中途取消还是发生异常,都返回true。future.isDone()
boolean isCanceller() :如果任务完成前被取消,则返回true。 future.isCanceller()
boolean cancel(boolean mayInterruptRunning) :如果任务还没开始,执行cancel(...)方法将返回false;如果任务已经启动,执行cancel(true)方法将以中断执行此任务线程的方式来试图停止任务,如果停止成功,返回true;当任务已经启动,执行cancel(false)方法将不会对正在执行的任务线程产生影响(让线程正常执行到完成),此时返回false;当任务已经完成,执行cancel(...)方法将返回false。mayInterruptRunning参数表示是否中断执行中的线程。
通过方法分析我们也知道实际上Future提供了3种功能:
(1)能够中断执行中的任务
(2)判断任务是否执行完成
(3)获取任务执行完成后额结果。
Future模式
Future模式的核心在于:去除了主函数的等待时间,并使得原本需要等待的时间段可以用于处理其他业务逻辑
Futrure模式:对于多线程,如果线程A要等待线程B的结果,那么线程A没必要等待B,直到B有结果,可以先拿到一个未来的Future,等B有结果是再取真实的结果。
在多线程中经常举的一个例子就是:网络图片的下载,刚开始是通过模糊的图片来代替最后的图片,等下载图片的线程下载完图片后在替换。而在这个过程中可以做一些其他的事情。
首先客户端向服务器请求RealSubject,但是这个资源的创建是非常耗时的,怎么办呢?这种情况下,首先返回Client一个FutureSubject,以满足客户端的需求,于此同时呢,Future会通过另外一个Thread 去构造一个真正的资源,资源准备完毕之后,在给future一个通知。如果客户端急于获取这个真正的资源,那么就会阻塞客户端的其他所有线程,等待资源准备完毕。