服务提供者框架是指:多个服务提供者实现一个服务,系统为客户端提供多个实现,并把他们从多个实现中解耦出来。服务提供者的改变对它们的客户端是透明的,这样提供了更好的可扩展性。例如,JDBC,JMS等就是用了服务提供者框架
他们之间的关系如下图:
1.服务具体实现类和服务提供者实现类是服务提供者自己去实现。以JDBC为例,这2个模块由具体的数据库提供商来实现。
2.其他三个模块是java对数据库提供商怎么实现上面2个模块的一个约束。比如:提供服务者实现类必须实现服务提供者接口。才能成功注册到服务提供者注册类。以JDBC为例,所有的数据库提供商只需要按照接口里面定义的规则来操作,都能成功地使java连上他们的数据库。
下面以mysql数据库为例简单说明一下mysql数据库提供商是如何实现这些接口的。
1.我们在java中获取mysql连接对象源代码如下:
1. Class.forName("com.mysql.jdbc.Driver");
2. 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元怎么处理等。一次性卡则没必要了。
服务接口源码:
1. /**
2. * 进出地铁服务接口
3. * @author Administrator
4. *
5. */
6. public interface
7.
8. //进入地铁
9. public boolean
10.
11. //出地铁
12. public boolean
13.
14. }
服务实现源码:
1. /**
2. * 一卡通地铁进出服务实现
3. * @author Administrator
4. *
5. */
6. public class SubWayImpl implements
7.
8. public boolean
9. "通过一卡通进入地铁");
10. /**
11. * 进行一些处理,然后返回是否放行
12. */
13. return false;
14. }
15.
16. public boolean
17. "通过一卡通出地铁");
18. /**
19. * 进行一些处理,然后返回是否放行
20. */
21. return false;
22. }
23.
24. }
服务提供者接口源码:
1. /**
2. * 地铁进出服务提供者接口
3. * @author Administrator
4. *
5. */
6. public interface
7.
8. public
9.
10. }
服务提供者接口实现源码:
1. /**
2. * 服务提供者实现类
3. * @author Administrator
4. *
5. */
6. public class SubwayProviderImpl implements
7.
8. static
9. "一卡通", new
10. }
11.
12. public
13. return new
14. }
15.
16. }
服务提供者注册类实现源码:
1. /**
2. * 服务提供者注册类
3. *
4. * @author Administrator
5. *
6. */
7. public class
8.
9. private
10.
11. }
12.
13. private static final Map<String, SubwayProviderInterface> providers = new
14.
15. public static void
16. providers.put(name, p);
17. }
18.
19. public static
20.
21. SubwayProviderInterface p = providers.get(name);
22.
23. if (p == null) {
24. throw new
25. "No provider registered with name:"
26. }
27.
28. return
29.
30. }
31.
32. }
客户端测试类:
1. /**
2. * 客户端测试类
3. * @author Administrator
4. *
5. */
6. public class
7.
8. /**
9. * @param args
10. * @throws ClassNotFoundException
11. */
12. public static void main(String[] args) throws
13.
14. "cn.netjava.cgl.subway.SubwayProviderImpl");
15. "一卡通");
16. swi.in();
17. swi.out();
18. }
19. }
测试类Class.forName("")里面的参数你可能跟我的不同,看你服务提供者实现类放在哪个包下面了。我的是在:cn.netjava.cgl.subway包下面。