一步步学习java并发编程模式之Active Object模式(三)java实现active object

      现在我们已经知道active object模式的核心思想,也明白了如何自己写一段实现类似效果的java代码。现在我们按照active object模式的角色分工,将第二篇中的例子重新实现下,参考了Active Object并发模式在Java中的应用 这篇博客。在 Active Object 模式中,主要有以下几种类型的参与者:

  • 代理 (proxy) :代理是 Active Object 所定义的对于调用者的公共接口。运行时,代理运行在调用者线程的上下文中,负责把调用者的方法调用转换成相应的方法请求 (Method Request),并将其插入相应的 Activation List,最后返回给调用者 Future 对象。代理角色具有以下几个特点:
        1.proxy和servant有着相同的接口签名,或者proxy比servant接口更简单好用。这样可以方便调用者无差别的对待proxy和servant,或           者更容易使用代理
        2.创建client需要的future对象,并快速返回,避免调用者线程长时间等待.客户端可以通过future对象获取实际的运算结果
        3.将调用者的方法调用转换成相应的方法请求 (Method Request),将request和servant传给scheduler进行调度执行
        4.代理是运行在调用者线程中的,它对原始的耗时服务进行了封装,让客户端无差异的使用代理和真实对象,除了更快.
  • 方法请求(method request):方法请求定义了方法执行所需要的上下文信息,诸如调用参数等。
  • activation list:负责存储所有由代理创建的,等待执行的方法请求。从运行时来看,Activation List 会被包括调用者线程及其 Active Object 线程并发存取访问,所以,Activation List 实现应当是线程安全的。
  • 调度者 (scheduler):调度者运行在 Active Object 线程中,调度者来决定下一个执行的方法请求,而调度策略可以基于很多种标准,比如根据方法请求被插入的顺序 FIFO 或者 LIFO,比如根据方法请求的优先级等等。
  • servant::Servant 定义了 Active Object 的行为和状态,它是 Proxy 所定义的接口的事实实现。
  • future:调用者调用 Proxy 所定义的方法,获得 Future 对象。调用者可以从该 Future 对象获得方法执行的最终结果。在真实的实现里,Future 对象会保留一个私有的空间,用来存放 Servant 方法执行的结果。

我们现在先编写servant类,这个是整个active object角色中最简单的

package activeobject.aty.servant;

import activeobject.aty.result.WeatherResult;

//Servant 的实现是纯粹的应用逻辑实现,或者称为商业逻辑实现,没有混合任何的线程同步机制 , 这有利于我们进行应用逻辑的重用,而不需要考虑不同的线程同步机制。
public class WeatherServant
{
    public WeatherResult getWeatherInfo(String city, String day)
    {
        try
        {
            // 模拟耗时操作,调用该方法的线程,要挂起5s
            Thread.sleep(5 * 1000);
        }
        catch (InterruptedException e)
        {

        }

        WeatherResult result = new WeatherResult();
        result.setTemperature(28);
        result.setWindDirection("西北风");
        result.setDampness("74%");

        return result;
    }

}

package activeobject.aty.result;

// 模拟天气预报的查询结果
//今日天气实况:气温:4℃;风向/风力:西北风 1级;湿度:74%;空气质量:中;紫外线强度:最弱
public class WeatherResult
{
 // 气温
 private int temperature;

 // 湿度
 private String dampness;

 // 风向
 private String windDirection;

 public int getTemperature()
 {
     return temperature;
 }

 public void setTemperature(int temperature)
 {
     this.temperature = temperature;
 }

 public String getDampness()
 {
     return dampness;
 }

 public void setDampness(String dampness)
 {
     this.dampness = dampness;
 }

 public String getWindDirection()
 {
     return windDirection;
 }

 public void setWindDirection(String windDirection)
 {
     this.windDirection = windDirection;
 }

 @Override
 public String toString()
 {
     return "WeatherResult [temperature=" + temperature + ", dampness=" + dampness + ", windDirection="
             + windDirection + "]";
 }
 


}

现在编写methodrequest和future代码
package activeobject.aty.future;

import activeobject.aty.result.WeatherResult;

// 调用者调用 Proxy 所定义的方法,获得 Futur 对象。调用者可以从该 Future 对象获得方法执行的最终结果。
// 在真实的实现里,Future 对象会保留一个私有的空间,用来存放 Servant 方法执行的结果。
public class WeatherFuture
{
    private boolean isDone = false;

    private WeatherResult result = null;

    public WeatherResult get()
    {
        while (!isDone)
        {

        }

        return result;
    }

    public void setDone(boolean isDone)
    {
        this.isDone = isDone;
    }

    public void setResult(WeatherResult result)
    {
        this.result = result;
    }

}

package activeobject.aty.methodrequest;

import activeobject.aty.future.WeatherFuture;
import activeobject.aty.result.WeatherResult;
import activeobject.aty.servant.WeatherServant;

// Method Request:方法请求定义了方法执行所需要的上下文信息,诸如调用参数、返回结果和实际的服务等
public class WeatherMethodRequest
{
    private WeatherFuture future;

    private WeatherServant servant;

    private String city;

    private String day;

    public WeatherMethodRequest(WeatherFuture future, WeatherServant servant, String city, String day)
    {
        this.future = future;
        this.servant = servant;
        this.city = city;
        this.day = day;
    }

    public void call()
    {
        WeatherResult result = servant.getWeatherInfo(city, day);

        future.setResult(result);
        future.setDone(true);
    }

}

下面编写activation list
package activeobject.aty.activationlist;

import java.util.ArrayList;
import java.util.List;

import activeobject.aty.methodrequest.WeatherMethodRequest;

// Activation List 负责存储所有由代理创建的,等待执行的方法请求。
// 从运行时来看,Activation List 会被包括调用者线程及其 Active Object线程并发存取访问,所以Activation List 实现应当是线程安全的.

// Activation List 的实际上就是一个线程同步机制保护下的 Method Request队列,对该队列的所有操作 (insert/remove)都应该是线程安全的。
// 从本质上讲,Activation List 所基于的就是典型的生产者 / 消费者并发编程模型,调用者线程作为生产者把
// Method Request放入该队列,Active Object 线程作为消费者从该队列拿出 Method Request, 并执行。
public class WeatherActivationList
{
    private List<WeatherMethodRequest> requestList = new ArrayList<WeatherMethodRequest>();

    public synchronized void insertTask(WeatherMethodRequest request)
    {
        requestList.add(request);
    }

    public synchronized void removeTask(WeatherMethodRequest request)
    {
        requestList.remove(request);
    }

    public synchronized boolean isEmpty()
    {
        return requestList.size() == 0;
    }

    public synchronized WeatherMethodRequest popFirst()
    {
        WeatherMethodRequest e = requestList.get(0);

        requestList.remove(0);

        return e;
    }
}

到这里我们的请求已经在activatio list中了,我们可以基于activation list编写我们的scheduler实现
package activeobject.aty.scheduler;

import activeobject.aty.activationlist.WeatherActivationList;
import activeobject.aty.methodrequest.WeatherMethodRequest;

// Active Object 的线程
public class WeatherStartThread implements Runnable
{
    private WeatherActivationList safeRequestList;

    public WeatherStartThread(WeatherActivationList safeRequestList)
    {
        this.safeRequestList = safeRequestList;
    }

    @Override
    public void run()
    {
        while (true)
        {
            if (!safeRequestList.isEmpty())
            {
                WeatherMethodRequest request = safeRequestList.popFirst();
                request.call();
            }
        }
    }

}

package activeobject.aty.scheduler;

import activeobject.aty.activationlist.WeatherActivationList;
import activeobject.aty.methodrequest.WeatherMethodRequest;

// scheduler:调度者运行在 Active Object线程中,调度者来决定下一个执行的方法请求,
// 而调度策略可以基于很多种标准,比如根据方法请求被插入的顺序 FIFO 或者 LIFO,比如根据方法请求的优先级等等。
public class WeatherTaskScheduler
{
    private WeatherActivationList safeRequestList = new WeatherActivationList();

    public WeatherTaskScheduler()
    {
        new Thread(new WeatherStartThread(safeRequestList)).start();
    }

    public void insertRequest(WeatherMethodRequest methodRequest)
    {
        safeRequestList.insertTask(methodRequest);
    }


}

下面我们来编写代理类的实现,这个是客户端直接使用的类
package activeobject.aty.proxy;

import activeobject.aty.future.WeatherFuture;
import activeobject.aty.methodrequest.WeatherMethodRequest;
import activeobject.aty.scheduler.WeatherTaskScheduler;
import activeobject.aty.servant.WeatherServant;

//代理 (Proxy)负责以下功能:
// 1.proxy和servant有着相同的接口签名,或者proxy比servant接口更简单好用。这样可以方便调用者无差别的对待proxy和servant,或者更容易使用代理
// 2.创建client需要的future对象,并快速返回,避免调用者线程长时间等待.客户端可以通过future对象获取实际的运算结果
// 3.将调用者的方法调用转换成相应的方法请求 (Method Request),将request和servant传给scheduler进行调度执行
// 4.代理是运行在调用者线程中的,它对原始的耗时服务进行了封装,让客户端无差异的使用代理和真实对象,除了更快.
public class WeatherProxy
{
    // 对任务进行调度
    private static WeatherTaskScheduler scheduler = new WeatherTaskScheduler();

    // 服务的真正实现
    private static WeatherServant servant = new WeatherServant();

    public static WeatherFuture getWeatherInfo(String city, String day)
    {
        WeatherFuture futureResult = new WeatherFuture();

        WeatherMethodRequest request = new WeatherMethodRequest(futureResult, servant, city, day);

        scheduler.insertRequest(request);

        return futureResult;
    }
}

最后我们编写测试类,来验证我们的功能。
package activeobject.aty;

import activeobject.aty.future.WeatherFuture;
import activeobject.aty.proxy.WeatherProxy;
import activeobject.aty.result.WeatherResult;

public class TestWeather {

	public static void main(String[] args) throws Exception {
		mockMultiCall();

		testAsyncCall();
	}

	public static void testAsyncCall() throws Exception {
		// 1.调用天气计算服务,开始计算深圳的天气情况(开始计算)
		WeatherFuture future = WeatherProxy
				.getWeatherInfo("广东深圳", "2014-03-14");

		// 2.后台在计算天气服务中.当前线程没有阻塞,仍然可以继续执行.
		System.out.println("After calling weather service,i am still running.");

		// 3.The current thread is not blocked, do something else here...
		// Thread.sleep(5 * 1000);

		// 4.与天气计算结果无关的代码执行完毕.
		System.out.println("Now,i really need weather result to continue.");

		// 5.如果计算天气还没有结束,那么当前线程挂起,等候计算完成.
		WeatherResult info = future.get();
		System.out.println("天气查询结果:" + info);
	}

	public static void mockMultiCall() {
		Thread t = new Thread(new Runnable() {
			@Override
			public void run() {
				while (true) {
					long time = System.currentTimeMillis();

					System.out.println("构造一个服务调用请求:");
					WeatherFuture future = WeatherProxy.getWeatherInfo("广东深圳"
							+ time, "2014-03-14");
					WeatherResult info = future.get();
					System.out.println("mockMultiCall结果:" + info);
					System.out.println();
					System.out.println();
				}
			}
		});

		t.start();
	}
}

这样我们就按照active object模式的各个参与者及职责完成了java实现。

一步步学习java并发编程模式之Active Object模式(三)java实现active object,布布扣,bubuko.com

一步步学习java并发编程模式之Active Object模式(三)java实现active object

上一篇:2013级C++第4周(春)项目——再和对象找感觉


下一篇:Java 操作 properties 文件