博主是一名苦逼的大四实习生,现在java从业人员越来越多,面对的竞争越来越大,还没走出校园,就TM可能面临失业,而且对那些增删改查的业务毫无兴趣,于是决定提升自己,在实习期间的时间还是很充裕的,期间自学了很多流行的技术(我差点信了),也重拾了之前忽略的数据结构和算法(虽然还是半吊子,总比一点不会好,哈哈),欢迎大家和我交流,特别是想提升自身技能的小伙伴,我的QQ是673793576,下面开始进入正题~~

以前我看书的时候都是走马观花的看,看完以为都懂了,其实毛都不会,纯属浪费时间,最近在看2本书,effective java 和 java 并发编程实战,两本书绝对是难得的好书,强烈推荐大家去仔细精读,前提是有java基础,不放过书中提到的每一个知识点,不好意思,又写了一段与正题无关的内容,哈哈。。。。

服务提供者框架其实我们并不陌生,从最开始的jdbc编程中,我们就已经遇见了它,常用的数据库有MySQL数据库,Oracle数据库等,这些数据库由于是不同的数据库提供方提供,导致在java连接数据库的连接方式也不一样,这里我们可以把这个连接看成一个服务,作为开发人员,我们只管去调用这个服务,而不需要管这个服务是怎么产生的,服务的生成由这些服务提供者去产生,我们要做的就是指定一个服务提供者的名字,然后就可以得到服务,而无需关注具体实现细节,那么问题来了,我们要获取服务,从哪里获取,直接向服务提供者索要吗?通常的做法是,用一个类似管家的东西来集中管理这些服务的提供者,作为服务消费方,我们直接去找这个管家索要服务

服务提供者框架的几个概念:服务接口以及具体的服务实现类,服务提供者接口(可选)以及具体的实现类,服务提供者的注册类(也就是上文所说的管家)

服务是一个很抽象的概念,既然是服务,它肯定有个作用啊,对应到java中,它应该具有方法,事实上,服务的种类有很多,学过java的都知道,我们可以用一个接口来定义一个服务,然后让不同的实现类去实现这个接口从而实现具体的服务内容。

服务接口:定义了抽象的服务,以及一个或多个方法

服务实现类:实现服务接口,实现具体的方法

服务提供者接口:定义了抽象的服务提供者,提供一个获取服务的方法

服务提供者实现类:实现服务提供者接口,实现具体的方法

服务提供者注册类:提供一个注册服务与调用服务的方法,它是面向服务消费者的

废话不说直接上图

接着上代码

 package serviceprovider;

 public interface Service {    

     void print();
}
package serviceprovider;

public class FirstService implements Service {

    @Override
public void print() {
System.out.println("第一个服务");
} }
package serviceprovider;

public class SecondService implements Service {

    @Override
public void print() {
System.out.println("第二个服务");
} }
package serviceprovider;

public interface ServiceProvider {
Service getService(); }
package serviceprovider;

public class FirstServiceProvider implements ServiceProvider {

    static{
ServiceManager.registerProvider("firstServiceProvider", new FirstServiceProvider());
} @Override
public Service getService() {
return new FirstService();
} }
package serviceprovider;

public class SecondServiceProvider implements ServiceProvider {

    static{
ServiceManager.registerProvider("secondServiceProvider", new SecondServiceProvider());
} @Override
public Service getService() {
return new SecondService();
} }
package serviceprovider;

import java.util.Map;
import java.util.concurrent.ConcurrentHashMap; public class ServiceManager { private static final Map<String, ServiceProvider> providers = new ConcurrentHashMap<String, ServiceProvider>(); public static void registerProvider(String name, ServiceProvider p) {
providers.put(name, p);
} public static Service getService(String name) {
ServiceProvider p = providers.get(name);
if (p == null) {
throw new IllegalArgumentException(
"No ServiceProvider registered with name:" + name);
}
return p.getService();
}
}
package serviceprovider;

public class Test {

    public static void main(String[] args) throws ClassNotFoundException {
Class.forName("serviceprovider.FirstServiceProvider");
Service service = ServiceManager.getService("firstServiceProvider");
service.print();
} }

执行结果:第二个服务

代码分析:首先定义一个名叫Service的服务,这里只提供一个print()方法,它有两个具体的实现类,第一个实现类的print方法打印出“第一个服务”内容,第二个同理,现在有了服务,还缺服务的提供者,也就是这些服务由谁来提供,同理定义一个提供者接口,既然是服务提供者,总得存在服务吧,没有服务尼玛还提供毛线啊,所以要持有一个取得服务的方法,这里用到了一个静态代码块,是为了提供者在类在初始化时向管理类注册自己,意思就是,管家(提供者管理类),我是XX,我提供XX服务,给我登记一下我的信息。第二个同理。然后来一个测试类,这里看到Class.forName是不是很熟悉,Class.forName("xxx"),会加载这个类的信息,并初始化静态代码块,因为静态对象属于类嘛,但是不会实例化这个对象,除非加上.newInstance()方法,即Class.forName("xxx").newInstance();这里用的是反射,就不细说了。所以在提供者类中用静态代码块想管理类注册自己(这样只需要Class.forName("xxx")就可以执行xxx里的静态内容了,对应到提供者类中就是实例化自身,放在管理类的map中去管理),还有一个管理类,作为服务调用方,主要就和这个管理类打交道,这个类用了一个ConcurrentHashMap集合来维护这些提供者信息,ConcurrentHashMap是一个线程安全的map,具体请自行百度,当有服务提供者注册时就put一个进去,当有调用方调用服务时就get一个出来,也就是key value的操作。

很简单吧~

再看看jdbc是怎么操作的。

Class.forName("com.mysql.jdbc.Driver");

这里的Driver就是我们的提供者类。负责提供Connection(服务)的

Connection connection= DriverManager.getConnection("jdbc:mysql:///xxx", "root", "root");

这里的DriverManager对应提供者管理类,它对外提供一个获取服务(也就是Connection)的方法。

jdbc:mysql说明这个服务提供者是MySQL的,那么从集合(这里用的不是map,是一种list数组集合,可以看看DriverManager类的源码)里取出实现MySQL数据库连接的服务提供者,调用服务提供者的获取服务方法(前文说过,服务提供者肯定有一个获取服务的方法),然后返回我们需要的服务,也就是Connection对象

完~

最新文章

  1. AFN3.0封装
  2. 神经网络及其PID控制
  3. 简单理解Struts2中拦截器与过滤器的区别及执行顺序
  4. 数据结构和算法 &ndash; 6.构建字典: DictionaryBase 类和 SortedList 类
  5. [.ashx檔?泛型处理例程?]基础入门#1....能否用中文教会我?别说火星文?
  6. JAVA WEB 作用域之间的区别
  7. 如何设置win7系统的文件夹为系统文件,从而隐藏文件夹
  8. oracle还原数据库及遇到的问题
  9. Android] Android XML解析学习——方式比较
  10. Spring切面通知执行的顺序(Advice Order)
  11. STM32使用cube生成的程序后在keil5编译后首次SWD可以下载再次下载不行的解决办法。
  12. https和http有什么区别
  13. 18 Ui美化
  14. kafka原理和架构
  15. erlang开发工具之intellij idea基本使用
  16. 离线安装Eclipse插件-Vrapper
  17. Python_list部分功能介绍
  18. 第二阶段第二次spring会议
  19. Python 3 利用 Dlib 19.7 进行人脸检测
  20. 【UGUI】 (二)--------- 小地图

热门文章

  1. 一款基于vue2.0的分页组件---写在页面内
  2. firefox被hao123绑架的解决办法
  3. 基于树莓派的智能家居项目的设想与实现 Hestia
  4. 你好 JSONP !!!!
  5. C语言之计算log2
  6. 浅谈IM(InstantMessaging) 即时通讯/实时传讯【理论篇】
  7. 使用docker搭建Jenkins 及slave的配置
  8. ThinkPHP中ajax绑定select下拉框无法显示
  9. cinder块存储控制节点
  10. nignx 测试配置文件