服务提供者框架中有三个重要组件:服务接口(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 Mapproviders = new ConcurrentHashMap (); /** * @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 Mapproviders = new ConcurrentHashMap (); /** * @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的实现。