服务提供框架:
服务接口:提供者实现,用于提供服务支持。例如:JDBC接口Connection。
提供者注册API:
服务提供框架用来注册实现,让客户端访问它们。 例如:DriverManager.registerDriver()。
服务访问API:
客户端用来获取服务的实例,从而调用服务接口。服务访问API一般允许但是不要求客户端指定某种选择提供者的条件,如果没有这样的规定,API就会返回默认实现的一个实例。例如:DriverManager.getConnection()。
服务提供者接口:
这些提供者负责创建其服务实现的实例。例如:Driver。
如果没有服务提供者接口,实现就按照类名称进行注册,通过反射的方式进行实例化。例如:注册JDBC驱动时jdbc.driver=com.mysql.jdbc.Driver,假设没有服务提供者接口,则提供者注册API会将该驱动类名信息储存,在调用服务访问API时,反射的方式进行实例化mysql驱动实例,从而访问服务接口方法。
Driver driver = new com.mysql.jdbc.Driver(); // 服务提供者接口
DriverManager.registerDriver("com.mysql.jdbc.Driver"); // 当然不推荐使用这种方式加载驱动,优先选择forName方式,在此只是举例阐明提供者注册API
Connection conn = DriverManager.getConnection(url); // 服务访问API
Statement stmt = conn.createStatement(); // 服务接口
服务提供者框架模式有无数变体,例如:服务访问API可以使用适配器模式,返回比提供者需要的更加丰富的服务接口。下面是一个简单实现:
// Service provider framework sketch
// Service interface
public interface Service {
...
// Service specific methods
}
// Service provider interface
Service interface Provider {
Service newService();
}
public class Services {
private Service(); // Prevents instantiation
private static final Map<String, Provider> providers = new ConcurrentHashMap<String, Provider>();
public static final String DEFAULT_PROVIDER_NAME = "<def>";
// Provider registration API
public static void registerDefaultProvider(Provider p) {
registerProvider(DEFAULT_PROVIDER_NAME, p);
}
public static void registerProvider(String name, Provider p) {
providers.put(name, p);
}
// Service access API
public static Service newInstance() {
return newInstance(DEFAULT_PROVIDER_NAME);
}
public static Service newInstance(String name) { // 静态工厂方法,跟设计模式中的静态工厂是不同概念
Provider provider = providers.get(name);
if (provider == null) {
throw new IllegalArgumentException("No provider registered with name: " + name);
}
return provider.newService();
}
}