他们之间的关系如下图:
1.服务具体实现类和服务提供者实现类是服务提供者自己去实现。以JDBC为例,这2个模块由具体的数据库提供商来实现。
2.其他三个模块是java对数据库提供商怎么实现上面2个模块的一个约束。比如:提供服务者实现类必须实现服务提供者接口。才能成功注册到服务提供者注册类。以JDBC为例,所有的数据库提供商只需要按照接口里面定义的规则来操作,都能成功地使java连上他们的数据库。
下面以mysql数据库为例简单说明一下mysql数据库提供商是如何实现这些接口的。
1.我们在java中获取mysql连接对象源代码如下:
- Class.forName("com.mysql.jdbc.Driver");
- DriverManager.getConnection("jdbc:mysql://localhost:3306/test","root","123");
从这个地方可以看出:我们只需要更换数据库的驱动名称与建立连接的URL,用户名等信息,就可以完全切换到另外一个数据库。数据库底部怎么操作的我们不清楚,也没必要清楚。我们获取的连接对象是Connection,查看java.sql.Connection这个类,会发现它只是一个接口。我们得到的只是一个接口,怎么可能能够操作数据库呢?其实这里得到的不是Connection接口,而是它的一个实现类,只是对于客户端不可见而已。这可能就是所谓的面向接口编程,客户端只需要知道它该知道的信息,服务端告诉客户端,你可以调用哪些方法。至于具体方法怎么实现是服务端的事情,客服端就不需要管,也不需要知道了。
下面我们看看简单的这2个语句分别做了什么事情:
语句一: Class.forName("...")。这样一个语句会实例化一个com.mysql.jdbc.Driver类(提供服务者实现类),并将这个类的实例注册到DriverManager(服务提供者注册类)。
语句二: 通过建立连接的URL,用户名,密码来获取建立到mysql数据库的连接。是这样的,DriverManager通过你传进来的url信息判断出你是要获取那个服务提供者提供的服务。也就是语句一已经将提供服务者实现类注册到DriverManager了,DriverManager获取到这个服务提供者实现类对象之后,通过调用它的getService(mysql里面是connect方法)方法获取到服务具体实现类对象,返回的却是java.sql.Connection接口对象(因为服务具体实现类实现了Connection接口),这样把服务具体实现类对象隐藏了。提供了很好的扩展性。
最后,我们自己来测试一个。
以北京地铁进出控制为例:现在北京地铁进出都是刷卡,有二种卡:1.一卡通(比如一次性冲值50元,进地铁刷一次,出地铁刷一次,扣2元)。2.一次性卡(进地铁刷一次,出地铁插入回收。)这2种卡都可以实现进出地铁功能,但实现的具体方法是有区别的:一卡通:需要获取这卡余额是多少,然后扣掉2元。如果余额不足2元怎么处理等。一次性卡则没必要了。
服务接口源码:
- /**
- * 进出地铁服务接口
- * @author Administrator
- *
- */
- public interface SubWayInterface {
- //进入地铁
- public boolean in();
- //出地铁
- public boolean out();
- }
服务实现源码:
- /**
- * 一卡通地铁进出服务实现
- * @author Administrator
- *
- */
- public class SubWayImpl implements SubWayInterface {
- public boolean in() {
- System.out.println("通过一卡通进入地铁");
- /**
- * 进行一些处理,然后返回是否放行
- */
- return false;
- }
- public boolean out() {
- System.out.println("通过一卡通出地铁");
- /**
- * 进行一些处理,然后返回是否放行
- */
- return false;
- }
- }
服务提供者接口源码:
- /**
- * 地铁进出服务提供者接口
- * @author Administrator
- *
- */
- public interface SubwayProviderInterface {
- public SubWayInterface getService();
- }
服务提供者接口实现源码:
- /**
- * 服务提供者实现类
- * @author Administrator
- *
- */
- public class SubwayProviderImpl implements SubwayProviderInterface {
- static {
- ServiceManager.registerProvider("一卡通", new SubwayProviderImpl());
- }
- public SubWayInterface getService() {
- return new SubWayImpl();
- }
- }
服务提供者注册类实现源码:
- /**
- * 服务提供者注册类
- *
- * @author Administrator
- *
- */
- public class ServiceManager {
- private ServiceManager() {
- }
- private static final Map<String, SubwayProviderInterface> providers = new ConcurrentHashMap<String, SubwayProviderInterface>();
- public static void registerProvider(String name, SubwayProviderInterface p) {
- providers.put(name, p);
- }
- public static SubWayInterface getService(String name) {
- SubwayProviderInterface p = providers.get(name);
- if (p == null) {
- throw new IllegalArgumentException(
- "No provider registered with name:" + name);
- }
- return p.getService();
- }
- }
客户端测试类:
- /**
- * 客户端测试类
- * @author Administrator
- *
- */
- public class Test {
- /**
- * @param args
- * @throws ClassNotFoundException
- */
- public static void main(String[] args) throws ClassNotFoundException {
- Class.forName("cn.netjava.cgl.subway.SubwayProviderImpl");
- SubWayInterface swi = ServiceManager.getService("一卡通");
- swi.in();
- swi.out();
- }
- }
测试类Class.forName("")里面的参数你可能跟我的不同,看你服务提供者实现类放在哪个包下面了。我的是在:cn.netjava.cgl.subway包下面。