Future模式的适用场景
"想要得到数据时,先返回数据的包装,异步获取真实数据,真正需要时再拿数据,真实数据也许已经异步得到,也许还在阻塞获取中..."
换句话说,Future模式是非阻塞的,客户端请求服务端后,服务端只需要创建一个工作线程,然后马上返回了,而这个过程耗时很短,不会阻塞client,client在这期间可以去处理其他任务,当需要的时候向工作进程获取结果。
这有点像BOSS布置任务,比如由于市场需要,将在本月20号发布一个版本,今天是10号,分配任务给小弟去做了,询问小弟能不能搞定,由于小弟着急上位就答应了。而BOSS忙着处理每天的商务问题,在20号之前都没有空理会给小弟分配的任务。
不知不觉时间到了20号,这时BOSS赶忙询问小弟做的到底如何了,其实这个时候小弟还差10%没有搞定,但是不想影响自己以后的信誉就回答已经搞定了,准备晚上部署,BOSS拍了拍小弟肩膀说“好样的!过一段给你加薪!”,BOSS走开后,小弟立刻拿出自己吃奶的力气,最终在晚上部署前完成了任务。
思想就是发起请求后会立即返回一个数据给你,但是这个数据只是个包装,包装数据会异步请求真实数据,最终用的时候去拿真实的数据,可能异步加载完成直接返回,也可能未完成阻塞着。
再比如情人节到了,女朋友问你准备礼物了没有,明明忘记了,但是依然说准备了,然后偷偷赶紧网购个,对吧!起码事情算是圆满了。
使用wait和notify实现Future模式的工作原理
public interface Data {
String getRequest();
}
public class FutureData implements Data{
private RealData realData ;
private boolean isReady = false;
public Thread currentThread;
//装载真正数据
public synchronized void setRealData(RealData realData) {
//如果已经装载完毕了,就直接返回
if(isReady){
return;
}
//如果没装载,进行装载真实对象
this.realData = realData;
isReady = true;
//进行通知
notify();
}
@Override
public synchronized String getRequest() {
//如果没装载好 程序就一直处于阻塞状态
while(!isReady){
try {
wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
//装载好直接获取数据即可
return this.realData.getRequest();
}
}
public class RealData implements Data{
private String result ;
public RealData (String queryStr){
System.out.println("根据" + queryStr + "进行查询,这是一个很耗时的操作..");
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("操作完毕,获取结果");
result = "查询结果";
}
@Override
public String getRequest() {
return result;
}
}
public class FutureClient {
public FutureData request(final String queryStr){
//1 我想要一个代理对象(Data接口的实现类)先返回给发送请求的客户端,告诉他请求已经接收到,可以做其他的事情
final FutureData futureData = new FutureData();
//2 启动一个新的线程,去加载真实的数据,传递给这个代理对象
Thread t = new Thread(new Runnable() {
@Override
public void run() {
//3 这个新的线程可以去慢慢的加载真实对象,然后传递给代理对象
RealData realData = new RealData(queryStr);
futureData.setRealData(realData);
}
});
t.start();
futureData.currentThread = t;
return futureData;
}
}
主方法:
public class Main {
public static void main(String[] args) throws InterruptedException {
FutureClient fc = new FutureClient();
FutureData data = fc.request("请求参数");
System.out.println("请求发送成功!");
System.out.println("做其他的事情..."+"请求数据线程状态:"+data.currentThread.getState());
//经测试代码阻塞在这一行,等待获取真实数据,而 以上代码则实时执行
String result = data.getRequest();
System.out.println(result+"请求数据的线程状态:"+data.currentThread.getState());
}
}