每天一例多线程[day15]----Future模式原理剖析

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());
		
	}
}

 


 

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

jeff.sheng

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值