多线程之Future模式

  对于多线程,当A线程需要获得B线程的处理结果,而B线程处理业务需要很长时间,这时候A线程如果一直等待B线程的处理结果,线程A才能继续往下执行代码,这种方式在效率不是很好。所以,这种场景可以使用多线程的Future模式。

  Future模式,就是当A线程需要B线程的处理结果,启动B线程,然后A线程继续往下走,而不必一直等待B线程的结果,当A线程需要用到B线程的结果时候再去获取结果,如果B线程还没处理好业务逻辑,则A线程一直处于阻塞状态。直到B线程处理完成。下面总结一下Future模式的实现方式。

  首先客户端向服务器请求,但是这个资源的创建是非常耗时的,这种情况下,首先返回Client一个FutureSubject,以满足客户端的需求,同时,Future会通过另外一个Thread 去构造一个真正的资源,资源准备完毕之后,在给future一个通知。如果客户端急于获取这个真正的资源,那么就会阻塞客户端的其他所有线程,等待资源准备完毕。

  公共数据接口,FutureData和RealData都要实现

public interface Data {

    public abstract String getConent();
}
public class FutureData implements Data{

    private RealData realData;
    
    private boolean isReady=false;
    
  //进行真实数据的装载
public synchronized void setRealData(RealData realData){ if(isReady) return; isReady=true; this.realData=realData; notify(); } //如果业务还没处理完成,则进入阻塞状态 public synchronized String getConent() { while(!isReady){ try { wait(); } catch (InterruptedException e) { e.printStackTrace(); } }
    //业务逻辑处理完成,直接返回真实结果
return this.realData.getConent(); } }
public class RealData implements Data{

    private String result;
    //真正处理业务逻辑的方法
    public RealData(String queryStr){
        System.out.println("这是一个很耗时的请求...");
        try {
            Thread.sleep(5000);
        } 
        catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("操作完毕,获取结果");
        result="查询结果";
    }
    
    public String getConent() {

        return result;
    }

}
public class FutureClient {
    //处理客户端的请求
    public Data request(final String queryStr){
    
final FutureData futureDate =new FutureData();
     //开启一个新的线程去处理复杂的业务逻辑
new Thread(new Runnable() { public void run() { RealData realData=new RealData(queryStr);
         //处理完成对数据进行装载 futureDate.setRealData(realData); } }).start();
     //返回给客户端
return futureDate; } }
public class TestFuture {

    public static void main(String[]args){
        FutureClient fc=new FutureClient();
        Data data=fc.request("查询条件..");
        System.out.println("请求发送成功!");
        System.out.println("做其他的事情...");
        String result = data.getConent();
        System.out.println(result);
        
    }
}

  在JDK1.5 Concurrent 中,提供了这种Callable和Future。Callable产生结果,Future拿到结果。 代码如下:

public class Task implements Callable<String> {

    public String call() throws Exception {

        Thread.sleep(5000);
        
        return "Hello Future";
    }
}

  首先处理业务逻辑的类必须实现Callable接口,他与runable接口的区别是,runable接口重写的Run方法没有返回值,而Callable接口可以定义返回值。

public class FutureDemo {

    public static void main(String[]args) throws InterruptedException, ExecutionException{
        
         System.out.println("已经提交资源申请");
         
         ExecutorService executor=Executors.newCachedThreadPool();
         
         Task task =new Task();
         
         Future<String> future=executor.submit(task);
         
         System.out.println("继续做其他事情...");
         
         if(!future.isDone()){
             System.out.println("任务还没处理完成.."); 
         }
         System.out.println("获取返回结果:"+future.get());
         
         if(future.isDone()){
             System.out.println("任务处理完成"); 
         }
         executor.shutdown();
    }
}

  使用线程池 ExecutorService executor=Executors.newCachedThreadPool(),调用submit方法开启线程,submit方法与executor方法的区别是,submit可以传入实现Callable或者runable接口类型的参数,而executor只可以传入实现runable接口类型的参数。

posted @ 2017-04-23 19:29  gdpuzxs  阅读(321)  评论(0编辑  收藏  举报