java实现异步调用
在JAVA平台,实现异步调用的角色有如下三个角色:
调用者取货凭证真实数据
一个调用者在调用耗时操作,不能立即返回数据时,先返回一个取货凭证.然后在过一断时间后
凭取货凭证来获取真正的数据.
所以连结调用者和真实数据之间的桥梁是取货凭证.我们先来看它的实现:
publicclassFutureTicket{
privateObjectdata=null;
privatebooleancompleted=false;
publicsynchronizedvoidmakeRealData(){
if(this.complited)return;
//获取数据的耗时操作.这里用Sleep代替
try{
Thread.sleep(10000);
}catch(Throwablet){}
this.data="返回的数据内容";
this.completed=true;
notifyAll();
}
publicsynchronizedObjectgetData(){
while(!this.completed)){
try{
wait();
}catch(Throwablet){}
}
returnthis.data;
}
publicbooleanisCompleted(){
returnthis.completed;
}
}
为了简单化说明(不把它们的关系开得复杂),这里用Object代替了真实数据.而真实的实现中
我们应该把makeData放在一个真实数据的类中,然后提供一个方法返回真实数据.这样对于真实
数据的处理和取货凭证解耦.
对于这个取货凭证,调用者的如何调用是异步调用的关键:
publcclassRequester{
publicFutureTicketrequest(){
finalFutureTicketft=newFutureTicket();
//在新线程中调用耗时操作
newThread(){
publicvoidrun(){
ft.makeRealData();
}
}.start();
returnft;
}
}
在新线程中启动耗时操作后,不等待线程的完成立即返回提货单.
然后调用者可以根据ft.isCompleted()来调用getData()获取真实数据.
当然对ft.isCompleted()测试可以按规定时间间隔轮巡(极低级的方案),也可以
在条件不满足时wait(),然后等待makeData的notifyAll();这样你就完成了一个
用JAVA模拟的异步操作.
改进:
但这样的调用对于调用者来说仍然要继续控制线程操作.如果调用者是一个资深的
程序员,这当然没有问题.但假如我们把对直接数据的处理委托给取货凭证来做.调用
者直接规定对数据的操作,然后由取货凭证来调用规定的操作,这对于调用者是一个很
好的解脱:
interfaceProcessData{
publicvoidprocess(Onjectdata);
}
publicMyProcessData{
publicvoidprocess(Objectdata){
//你不管什么时候起初数据data被获取了.
//你只要规定如果获取到数据了如何处理
System.out.println(data.toString()+"处理完成...........");
//insertintodataBase?
}
}
取货凭证在接收调用者请求获取数据时,要知道对获取的数据如何处理的方法:
publicclassFutureTicket{
privateObjectdata=null;
privatebooleancompleted=false;
privateProcessDatapd;
publicFutureTicket(ProcessDatapd){
this.pd=pd;
}
publicsynchronizedvoidmakeRealData(){
if(this.complited)return;
//获取数据的耗时操作.这里用Sleep代替
try{
Thread.sleep(10000);
}catch(Throwablet){}
this.data="返回的数据内容";
this.completed=true;
notifyAll();
}
publicsynchronizedvoidprocessData(){
while(!this.completed)){
try{
wait();
}catch(Throwablet){}
}
//returnthis.data;
//不用返回了,直接处理
this.pd.process(this.data);
//alert(?);
}
//这个方法也可以不要了.
publicbooleanisCompleted(){
returnthis.completed;
}
}
调用:
finalFutureTicketft=newFutureTicket(newMyProcessData());
//在新线程中调用耗时操作
newThread(){
publicvoidrun(){
ft.makeRealData();
}
}.start();
ft.processData();
OK,你现在可以抽烟,喝酒,泡妞.ft会为你完成所有的工作.