服务提供者框架 《Effective Java 一》

引入:静态工厂方法返回的对象所属的类,在编写包含该静态工厂方法的类时可以并不存在,这种灵活的静态工厂方法构成了服务提供者框架的基础。——引自effectivejava
服务提供者框架中有三个重要组件:服务接口(Service Interface),由服务提供者实现;提供者注册API(Provider Registration API),这是系统用来注册实现,让客户端访问他们的;服务访问API(Service Access API),是客户端用来获取服务的实例的。

服务提供者框架的第四个组件是可选的:服务提供者接口(ServiceProviderInterface),这些提供者负责创建其服务实现的实例。如果没有服务提供者接口,实现就按照类名称注册,并通过反射方式进行实例化。--转

看似简单,不仔细想想也是不行的。服务提供者框架,我理解涉及到三方面角色,一个是系统服务,他只负责提供服务,却不提供具体的实现,比如交通部门,负责提供交通运输,但是他自己不提供具体的实现。具体服务的实现是其下层,提供者角色。他组要是提供具体的服务,而不对外公布其细节处理。第三个角色就是使用者,即客户端访问。根据客户端的不同需求,调用系统服务,系统服务交给具体的实现商实现。
框架的优势在于为提供商制定统一的接口规范,使其隐藏具体的实现细节。为客户端提供统一的访问接口,并根据需要灵活的切换不同的实现者。

系统服务
系统服务需要兼顾提供者和使用者,并且实现解耦。
对于供应商,系统可以定义好接口规范以供实现【服务提供者接口】:
package com.cs.aine.service;

public interface Provider {
	public Service getService(); 
}

提供接口之后,必须接受供应商的注册,不然供应商无法植入到系统服务中来【提供者注册API】:
package com.cs.aine.service;

import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

public class ServiceManager {
	public ServiceManager() {

	}

	private static final Map<String, Provider> providers = new ConcurrentHashMap<String, Provider>();

	/**
	 * @desc for provide to register provider
	 * @param name
	 * @param p
	 */
	public static void registerProvider(String name, Provider p) {
		providers.put(name, p);
	}

对于使用者,系统服务必须定义好统一的访问接口,这定义了系统具体提供了哪些服务【服务接口】:
package com.cs.aine.service;
/**
 * 
 * @author Aine
 *
 */
public interface Service {
	public void goDanYang();//系统提供了去丹阳的功能

	public String getFood();//系统提供了获取食物的功能
}

除此之外,系统服务必须公开访问服务的接口,否则客户端无法获取服务【服务访问API】:
package com.cs.aine.service;

import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

public class ServiceManager {
	public ServiceManager() {

	}

	private static final Map<String, Provider> providers = new ConcurrentHashMap<String, Provider>();

	/**
	 * @desc for client to get service
	 * @param name
	 * @return
	 */
	public static Service getService(String name) {

		Provider p = providers.get(name);

		if (p == null) {
			throw new IllegalArgumentException(
					"No provider registered with name:" + name);
		}

		return p.getService();
	}
}

提供者/供应商
提供者需要做两个事情:
1 实现系统服务的提供者接口,将自己注册到系统中去。
2 实现系统服务的服务接口,提供自己的实现,并且隐藏在框架之外,不对外公开
package com.cs.aine.provide;

import com.cs.aine.service.Provider;
import com.cs.aine.service.Service;
import com.cs.aine.service.ServiceManager;

public class BusService implements Provider{
	static {  
        ServiceManager.registerProvider("Bus", new BusService());  
    }  
	@Override
	public Service getService() {
		return new SerciceImpl();
	}
	/**
	 * 
	 * @author Aine
	 * @desc needn't internal class
	 */
	class SerciceImpl implements Service{

		@Override
		public void goDanYang() {
			System.out.println("haha, I'm Bus service. I can bring you to DanYang.");
		}

		@Override
		public String getFood() {
			System.out.println("haha, I'm Bus service. I can supply you bus food.");
			return "Bus Food";
		}
		
	}
}

客户端
客户端根据自己的需求,选择不同的服务实现者,比如去丹阳,你自己要选择好做什么交通工具,可以坐汽车,可以做火车,也可以做轮船等等,这都是客户端的事情。
package com.cs.aine.client;

import com.cs.aine.service.Service;
import com.cs.aine.service.ServiceManager;

public class ClientTest {
	public static void main(String [] args){
		try {
			Class.forName("com.cs.aine.provide.TrainService");
			Service service = ServiceManager.getService("Train");  
			service.goDanYang();
			String food = service.getFood();
			System.out.println("I hava food:"+food);
			
			Class.forName("com.cs.aine.provide.BusService");
			service = ServiceManager.getService("Bus");  
			service.goDanYang();
			food = service.getFood();
			System.out.println("I hava food:"+food);
		}  catch (ClassNotFoundException e) {
			e.printStackTrace();
		}
	}
}


运行结果
haha, I'm Train service. I can bring you to DanYang.
haha, I'm Train service. I can supply you train food.
I hava food:Train Food
haha, I'm Bus service. I can bring you to DanYang.
haha, I'm Bus service. I can supply you bus food.
I hava food:Bus Food


思考和讨论
1 在客户端必须要知道自己选择的服务提供商的java实现类必须有供应商的驱动包。
  客户端实例化的时候才会驱动注册服务,是否可以将注册服务依附在系统服务中,客户端就可以从服务框架中选择服务商了。
2 如何让框架提供default的实现。

你可能感兴趣的:(Effective Java)