服务提供者框架是指:多个服务提供者实现一个服务,系统为客户端提供多个实现,并把他们从多个实现中解耦出来。服务提供者的改变对它们的客户端是透明的,这样提供了更好的可扩展性。例如,JDBC,JMS等就是用了服务提供者框架

      他们之间的关系如下图:

  


 

   1.服务具体实现类和服务提供者实现类是服务提供者自己去实现。以JDBC为例,这2个模块由具体的数据库提供商来实现。

   2.其他三个模块是java对数据库提供商怎么实现上面2个模块的一个约束。比如:提供服务者实现类必须实现服务提供者接口。才能成功注册到服务提供者注册类。以JDBC为例,所有的数据库提供商只需要按照接口里面定义的规则来操作,都能成功地使java连上他们的数据库。

 

    下面以mysql数据库为例简单说明一下mysql数据库提供商是如何实现这些接口的。

    1.我们在java中获取mysql连接对象源代码如下:

 

   


JAVA集成OPENIM服务端框架 java 服务端框架_服务提供者


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元怎么处理等。一次性卡则没必要了。

     服务接口源码:

 

   


JAVA集成OPENIM服务端框架 java 服务端框架_服务提供者

1. /**
2.  * 进出地铁服务接口
3.  * @author Administrator
4.  *
5.  */
6. public interface
7.       
8. //进入地铁
9. public boolean
10.       
11. //出地铁
12. public boolean
13.   
14. }

 

 

   

   服务实现源码:

 

 


JAVA集成OPENIM服务端框架 java 服务端框架_服务提供者

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. }


 

 

    

   服务提供者接口源码:

 

 

  


JAVA集成OPENIM服务端框架 java 服务端框架_服务提供者

1. /**
2.  * 地铁进出服务提供者接口
3.  * @author Administrator
4.  *
5.  */
6. public interface
7.       
8. public
9.   
10. }


 

 

    服务提供者接口实现源码:

 

 

   


JAVA集成OPENIM服务端框架 java 服务端框架_服务提供者

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. }

 

   服务提供者注册类实现源码:

 

 

  


JAVA集成OPENIM服务端框架 java 服务端框架_服务提供者

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. }

 

 

     客户端测试类:

 

    


JAVA集成OPENIM服务端框架 java 服务端框架_服务提供者

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包下面。