`
frank-liu
  • 浏览: 1685302 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

ServiceLoader和DriverManager使用总结

 
阅读更多

前言

    这篇文章不是专门讲ServiceLoader的用法,这篇文章也不仅是讲DriverManager的用法。他们两个一个是java.util包里面的类,另外一个在jdbc里面应用的很多。从表面上看起来他们之间似乎没有多少的联系。实际上DriverManager对ServiceLoader的使用可以达到一种巧妙的效果。在这里我想探讨一下DriverManager使用到的一种设计思路以及对我们后续解决类似问题的指导。

ServiceLoader

    ServiceLoader是jdk6里面引进的一个特性。在第一次碰到的时候还是有点不太理解。从官方的文档来说,它主要是用来装载一系列的service provider。而且ServiceLoader可以通过service provider的配置文件来装载指定的service provider。

    相信看完前面的这一段描述,依然还是会一头雾水。那么我们就结合一个具体的示例来讲讲吧。假定我们要定义一个接口MessageService,它的定义如下:

 

package com.test.service;

public interface MessageService {
    String getMessage();
}
    这是一个再简单不过的接口了。我们这个接口本质上定义了一个行为规范,后续要遵守这个规范的类都需要满足它,也就是说要实现这个接口。针对我们不同的需要,我们可能有不同的实现。比如说我们有如下两种实现:

 

RawMessage:

 

package com.test.raw;
import com.test.service.MessageService;
public class RawMessage implements MessageService {
    public String getMessage() {
        return "Raw message";
    }
}
 FormattedMessage:

 

package com.test.format;

import com.test.service.MessageService;

public class FormattedMessage implements MessageService {
    public String getMessage() {
        return "Formatted message";
    }
}

    针对前面这个接口来说,我们这里的两个实现可以说是两个service provider。因为按照前面这个接口的规范,我们有两个具体的不同实现。他们也许对应着不同的情形。

    OK,这里我们理解了service provider的意思。那么有了这些个接口和service provider,他们有什么用呢?他们和Service Loader有什么关系呢?我们继续往下讨论。

    既然前面提到Service Loader是用来装载这些service provider的,那就是说我们可以使用它来将这些具体实现的类引入进来使用。我们在原来代码的目录下面创建一个META-INF/services的目录,并创建一个com.test.service.MessageService的文件。这个文件名比较有意思,它是和我们前面定义的MessageService类的全名一样的。然后我们在这个文件里保存如下的信息:

com.test.raw.RawMessage
com.test.format.FormattedMessage

    整个项目的结构以及相关配置文件的布局如下图所示:

     这里我们注意到,META-INF文件夹是放在src这个代码目录下。前面这部分的配置文件到底是做什么用的呢?他们为什么要命名成和类的全名一样?我们看一下使用他们的示例代码再来讨论。

我们在原来的项目中增加如下的代码:

package com.test.messageconsumer;

import java.util.ServiceLoader;

import com.test.service.MessageService;

public class MessageConsumer {

	public static void main(String[] args) {
		ServiceLoader<MessageService> serviceLoader = 
				ServiceLoader.load(MessageService.class);
		for(MessageService service : serviceLoader) {
			System.out.println(service.getMessage());
		}
	}

}

    这部分代码使用我们前面定义的类,它通过ServiceeLoader的load方法来装载MessageService类型的对象。我们在一个循环里去调用装载的service里的getMessage方法。如果运行这个程序,我们会发现一个如下的结果:

Raw message
Formatted message

    现在如果我们从结果来反推那些代码的作用,至少可以看到一点,我们这里的ServiceLoader起作用了。它这个load方法虽然是load MessageService类型的class,实际上把MessageService的所有子类型都装载了进来并可以使用。那么java是怎么知道我们有这么两个子类呢?再想想我们前面定义的META-INF下面的配置文件,我们就可以猜出来了。没错,java通过查找META-INF目录下对应MessageService全名的文件,然后把文件里所有的项读取出来然后尝试去装载它们。我们可以尝试一下去验证他们,比如说我们把com.test.service.MessageService文件里后面的com.test.format.FormattedMessage这一行给删除了再去运行程序,我们会发现只有如下的输出了:

Raw message

    可见,java也不是通过施了什么魔法就找到我们定义的这些类,它是通过读取这个配置文件才知道的。我们现在回过头来看看前面的代码。我们定义了一个接口和它的两个实现,然后设置好配置文件后,一个使用他们的程序可以通过读取配置文件来初始化具体的实现。在使用他们的代码里,我们直接操作的是MessageService这个接口,并不是具体的实现。从这一点来说,很符合我们面向对象中针对接口和抽象编程的原则。而且这个时候如果对依赖注入比较熟悉的人似乎也嗅到了这么一丝的味道。确实,这里可以说是一种依赖注入的一个简单实现,我们通过配置文件可以提供一些特定的类给使用程序。当然,这里针对ServiceLoader还有一个特定的限制,就是我们提供的这些具体实现的类必须提供无参数的构造函数,否则ServiceLoader就会报错。

ServiceLoader使用的思考

    目前来说,我们已经看到了ServiceLoader的一种简单使用场景。在前面这个示例中,我们对接口的实现是在同一个工程里面,如果我们需要使用他们的时候,完全没必要通过ServiceLoader再装载具体实现进来,我们完全可以通过一个ArrayList再将他们的具体实现一个个加进去就可以了。那么,什么时候用ServiceLoader比较合适呢?既然在同一个工程里我们jvm可以直接装载他们的实现,那么很可能就是我们要装载的实现不在同一个工程里,可能是需要我们动态添加的,这个时候,他们的引入不是编译时候加进来的,是在运行的时候加入的,我们不能像使用普通引入的静态类库那样来使用他们。所以这就是ServiceLoader的优点所在了。比如说我们有一组接口,有的实现是我们本地的,我们可以在使用代码里直接引入进来。而有的却是第三方实现的,他们可能会在运行的时候加入进来。那么我们事先是不清楚的,也就不可能定死了他们的实现是哪个具体的类名。在前面ServiceLoader的使用里,我不用把你具体的实现引用到代码里,而只是在配置文件里指定就可以了。这一点也是我们后面要讨论的DriverManager的一个重要的核心思想。

DriverManager的应用和设计思路

    DriverManager是jdbc里管理和注册不同数据库driver的工具类。从它设计的初衷来看,和我们前面讨论的场景有相似之处。首先一个,针对一个数据库 可能会存在着不同的数据库驱动实现。我们在使用特定的驱动实现时不希望修改现有的代码才能达到目的,而希望通过一个简单的配置就可以达到效果。

    比如说,我们现在有一个数据库的驱动A,我们希望在程序里使用它而不修改代码。一种理想的选择就是我们将驱动A的信息加入到一个配置文件中,程序通过读取配置文件信息将A加载进来。而以后如果我们希望改用另外一个驱动B的时候,我们只需要将配置文件里的信息修改成驱动B的。我们肯定不希望在代码里写什么registerDriver(new A());之类的代码。尤其在有的情况下我们根本没有使用这些驱动的源代码。

DriverManager的设计 

    下图是DriverManager相关的类图:

    在实际的应用中,我们每个具体的驱动都需要实现这里的Driver接口。比如说mysql jdbc的驱动,它的方法声明如下:

public class Driver extends NonRegisteringDriver implements java.sql.Driver

    从使用的角度来说,我们一般是通过如下的语句来实现装载的:

Class.forName(“com.mysql.jdbc.Driver”);

    看到这一步的时候,大家估计还是会有点疑惑,这里只是用的Class.forName方法,似乎和DriverManager没关系啊。没错,这一步从表面上看没什么关系,实际上我们来看他们具体的行为。Class.forName主要是做了什么呢?它主要是要求JVM查找并装载指定的类。这样我们的类com.mysql.jdbc.Driver就被装载进来了。而且在类被装载进JVM的时候,它的静态方法就会被执行。我们来看com.mysql.jdbc.Driver的实现代码。在它的实现里有这么一段代码:

static {
	try {
		java.sql.DriverManager.registerDriver(new Driver());
	} catch (SQLException E) {
		throw new RuntimeException("Can't register driver!");
	}
}

    很明显,这里使用了DriverManager并将该类给注册上去了。所以,对于任何实现前面Driver接口的类,只要在他们被装载进JVM的时候注册DriverManager就可以实现被后续程序使用。在实际项目中,如果我们需要有更大的装载灵活性,可以把Class.forName里的参数放到配置文件里,需要装载其他Driver的时候,改一下文件再重启一下程序就可以了。

    我们前面讨论的这种场景主要针对某一个驱动的加载。在某些情况下,我们可能会这样想,如果我们一个项目里要同时用到多个数据库呢?他们每个数据库甚至用到的驱动都不同。那么我们该怎么办呢?很显然,我们可以考虑将这些所有可用的驱动先加载进来,然后再根据不同的需要来选择。DriverManager里面就有这么一个方法可以事先加载已有的一些驱动。我们来看看DriverManager的一部分代码:

static {
        loadInitialDrivers();
        println("JDBC DriverManager initialized");
    }

    这部分代码是在DriverManager被加载进JVM执行的。而loadInitialDrivers()方法里有一个关键的实现部分如下:

AccessController.doPrivileged(new PrivilegedAction<Void>() {
            public Void run() {

                ServiceLoader<Driver> loadedDrivers = ServiceLoader.load(Driver.class);
                Iterator driversIterator = loadedDrivers.iterator();

                try{
                    while(driversIterator.hasNext()) {
                        println(" Loading done by the java.util.ServiceLoader :  "+driversIterator.next());
                    }
                } catch(Throwable t) {
                // Do nothing
                }
                return null;
            }
        });

    在这部分代码里,我们看到了一个熟悉的身影,没错,ServiceLoader。这部分代码将写到配置文件里的Driver都加载了进来。我们从前面ServiceLoader部分的介绍了解到,ServiceLoader.load()方法是将配置文件里指定的所有类都装载进来并调用他们的构造函数。而从前面com.mysql.jdbc.Driver的代码里我们就已经看到它们在被加载的时候又会调用DriverManager的registerDriver方法。这样将所有配置的jdbc driver注册和装载的过程就完成了。

    在DriverManager里面,有一个ArrayList,就专门用来保存注册好的Driver。它的定义如下:

private final static CopyOnWriteArrayList<DriverInfo> registeredDrivers = new CopyOnWriteArrayList<DriverInfo>();

    这里使用了CopyOnWriteArrayList是因为考虑到在多线程使用的环境下为了保证线程安全。immutable的设计思路,你懂的。

    现在,我们针对DriverManager的初始化和装载过程做一个梳理。首先,DriverManager在一开始的时候会尝试装载配置文件里设定的Driver以及系统环境属性里设定的Driver。作为那些被加载的Driver实现,他们本身在被装载时会在执行的static代码段里通过调用DriverManager.registerDriver()来把自身注册到DriverManager的registeredDrivers列表中。这样后面就可以通过得到的Driver来取得连接了。这种方式是JDBC4最新使用的方式。在之前使用的时候可以采用registerDriver和Class.forName的方法。为了保证不和特定的具体Driver耦合,采用Class.forName的方式会更好一点。不过现在连Class.forName的方式都省了。这样不管是显式还是隐式的加载驱动都保证这些驱动被保存到了DriverManager里的registeredDrivers列表里。

    我们再来看看对Driver的选择,这里用的Class.forName,它通过传入的String类型参数来确定具体的对象类型。虽然这里返回的只是一个Class对象。我们后面可以通过它的createInstance方法来创建具体的对象。这种手法也是很多工厂方法模式里惯用的手段。在这种方法里,我们通过创建好一系列的对象信息,然后根据传入的参数类型来创建具体的对象。但是返回的还是一个抽象类型。这样就完美的实现了和具体实现的解耦以及对象创建过程的封装。

    现在看来DriverManager实现所采用的思想和工厂模式有很多近似的地方。在我们一些具体问题的设计思路上也有可借鉴之处。下面针对项目中一个具体的问题来进行探讨。

一个具体问题的分析

    这是一个项目中碰到的具体的问题,问题的整体结构如下:

     我们的RequestProcessor从消息队列里边接收消息。然后根据发送过来的消息来选择不同的具体processor,比如说如果消息指定的是list server,那么我们就需要用listServerRequestProcessor。这样,根据不同的请求消息,我们就可以提供不同的processor。从这个步骤来说,我们可能就会考虑到一个典型的工厂模式应用。从问题具体的案例来说,我们消息队列发送过来的消息类型总共有20多个。如果根据这些消息类型去创建不同的processor对象的话,总共也就大概20多种。

    一种我们典型的做法就是将选择processor对象的代码放到一个方法里,然后用一堆的if, else判断条件的语句,像如下所示:

public RequestProcessor getProcessor(String messageType) {
    if(messageType == "createInstance") {
        return new CreateInstanceProcessor();
    } else if(messageType == "deleteInstance") {
        return new DeleteInstanceProcessor();
    } ...
//...
}

     这种做法不是不行,以后每次我们添加或者删除新类型的processor的时候还是很麻烦,需要来修改这边的代码。那么能不能不需要添加这么一堆的判断语句来实现选择对象的效果呢?这个时候,我们可以考虑借鉴DriverManager的思想了。

    首先一个我们可以将这些具体实现的request processor声明放到配置文件里,因为他们都继承自RequestProcessor,我们可以采用ServiceLoader把他们加载进来。下一个问题就是,我们该怎么来选择特定的request processor呢?我们可以学DriverManager里面getDriver的手法。在那里每个具体的Driver实现都提供了一个acceptsURL方法,这个方法判断是否符合该Driver的期望。比如说我给的url是com.mysql.jdbc.Driver,那么则只有mysql jdbc的Driver才能返回为true。我们在这里可以依葫芦画瓢的定义一个acceptMessage(String messageType)的方法。每个不同的实现根据不同的参数值来返回对象。那么我们前面那个方法就可以用如下的方式实现了。

public RequestProcessor getProcessor(String messageType) {
    for(ServiceLoader service : serviceList) {
        if(service.acceptMessage(messageType)) {
            return service;
        }
    }
    return null;
}

     我们这里的serviceList定义可以放在类的静态代码中:

ServiceLoader<MessageService> serviceList = 
				ServiceLoader.load(RequestProcessor.class);

    这是一种选择对象的方式,当然,我们也可以把所有对象放到一个map里面,然后根据传入的参数去选择。

总结 

    ServiceLoader是一个可以实现动态加载具体实现类的机制,通过它可以实现和具体实现代码的解耦。也可以实现类似于IOC的效果。在jdbc的DriverManager里,就使用了ServiceLoader的特性。对它这种特性的灵活运用可以带来很多代码上实现的简化。它也可以作为一种工厂方法实现的参考。

 

参考材料

effective java

http://docs.oracle.com/javase/7/docs/api/java/util/ServiceLoader.html

http://my.oschina.net/hanzhankang/blog/109794

http://ludaojuan21.iteye.com/blog/243528

  • 大小: 16.2 KB
  • 大小: 19 KB
  • 大小: 13.4 KB
分享到:
评论

相关推荐

    DriverManager 驱动加载管理程序

    DriverManager是一款专为驱动开发设计的实用工具,它集成了驱动加载、卸载以及系统核心结构信息的查看等功能,极大地简化了...使用DriverManager,开发者可以更加专注于驱动程序的逻辑设计,而无需担忧繁琐的底层操作。

    java数据库连接DriverManager

    Java中的`DriverManager`...总结,`DriverManager`在Java数据库连接中起着核心作用,负责驱动程序的管理和连接的建立,它是理解JDBC工作原理的基础。正确配置和使用`DriverManager`能够有效地实现与各种数据库的交互。

    GYWDK-DriverManager使用说明1

    GYWDK-DriverManager是一个专为Windows操作系统设计的动态链接库(DLL),它提供了一种接口来管理和操作驱动程序。该库主要包含两个导出函数,用于与驱动相关的操作,如安装、启动、停止和卸载。以下是这两个关键...

    通过DriverManager类提供的方法控制日志输出

    总结来说,虽然`DriverManager`类自身不直接提供日志控制,但我们可以通过调整数据库驱动的配置、使用Java内置的日志框架或者集成第三方日志库来实现日志输出的控制。理解这些机制有助于我们在开发过程中更好地监控...

    JDBC DriverManager.registerDriver(new Driver());

    总结来说,`JDBC DriverManager.registerDriver(new Driver())` 是一个重要的环节,它使得Java应用程序能够识别并使用特定数据库的驱动,从而实现数据库的连接。了解这一过程对于任何进行数据库操作的Java开发者都至...

    JDBCTM 指南:入门3 – DriverManager

    对于简单的应用程序,一般程序员需要在此类中直接使用的唯一方法是 DriverManager.getConnection。正如名称所示,该方法将建立与数据库的连接。JDBC 允许用户调用 DriverManager 的方法 getDriver、getDrivers 和 ...

    数据库jdbc驱动加载过程

    JDBC 4.0 版本引入了 Java SPI 机制来加载驱动程序,使用 ServiceLoader 机制来加载驱动程序,并将其注册到 DriverManager 中。这样,应用程序就可以使用 JDBC 驱动程序来连接到数据库并进行数据交互。

    驱动管家DriverManagerv1.0中文绿色免费版

    驱动管家DriverManager v1.0中文绿色免费版是一款专为用户设计的系统工具,主要功能集中在硬件驱动程序的管理上,包括驱动升级、驱动备份和驱动还原三个方面。这些功能对于保持计算机系统的稳定性和优化硬件性能至关...

    使用JDBC连接各种数据库总结

    在Java编程中,JDBC(Java Database Connectivity...JDBC的使用不仅限于这些基本步骤,还包括处理结果集、执行批处理、处理异常和优化性能等方面。了解并熟练掌握这些知识对于开发与数据库交互的Java应用程序至关重要。

    JDBC中驱动加载的过程分析(上)

    通过实现这个接口,驱动程序能够被`java.sql.DriverManager`类识别和管理,从而实现动态加载和使用。 #### 二、DriverManager类 `DriverManager`类是JDBC框架的核心组件之一,负责管理数据库驱动程序的注册和连接...

    JAVA与SQLServer数据库连接总结.doc

    本文将详细介绍JAVA与SQLServer数据库连接的总结,包括使用JDBC-ODBC桥方式和JDBC直接连接SQLServer数据库两种方法。 一、使用JDBC-ODBC桥方式连接SQLServer数据库 使用JDBC-ODBC桥方式连接SQLServer数据库需要...

    数据库基础学习的总结

    ### 数据库基础学习总结 #### 一、数据库概念与作用 **数据库**是一种组织和存储数据的方式,它允许用户高效地访问、管理和更新数据。在现代信息技术领域中,数据库是核心组成部分之一,广泛应用于各种软件系统和...

    常用的几种数据库连接方法总结

    最后,使用 DriverManager 获取连接对象。 代码示例: ```java Class.forName("Com.ibm.db2.jdbc.net.DB2Driver"); String url = "jdbc:db2://dburl:port/DBname"; cn = DriverManager.getConnection(url, sUsr, ...

    jdbc基础的详解与总结

    "jdbc 基础的详解与总结" jdbc 是一套协议,由 Sun 公司定义的一组接口,由数据库厂商来实现,并规定了 Java 开发人员访问数据库所使用的方法的调用规范。jdbc 的实现是由数据库厂商提供,以驱动程序形式提供。jdbc...

    JDBC知识点总结

    ### JDBC知识点总结 #### 一、JDBC概述 **简介** JDBC (Java Database Connectivity) 是由 SUN 公司提供的一套标准规范,用于在 Java 应用程序中访问关系型数据库。它允许 Java 程序通过统一的 API 与不同的...

    java,JDBC总结

    这些接口包括`DriverManager`、`Connection`、`Statement`、`PreparedStatement`和`CallableStatement`等。 2. 数据库驱动:JDBC驱动是Java程序与数据库之间的桥梁,分为四种类型:JDBC-ODBC桥、本地API驱动、网络...

    Java JDBC编程总结

    ### Java JDBC编程总结 #### 一、JDBC基本原理 JDBC(Java Database Connectivity)是一种用于执行SQL语句的Java API,它可以为多种关系数据库提供统一访问,这使得Java应用程序无需依赖于底层数据库的具体实现就...

    Java最全知识总结思维导图

    本资源"Java最全知识总结思维导图"提供了全面的Java学习路径和关键知识点的概览,旨在帮助初学者和有经验的开发者系统地理解和掌握Java的核心概念。 1. **数据库**: Java在数据库交互方面主要依靠JDBC(Java ...

    javaweb学习总结

    MySql-JDBC 的使用需要了解 JDBC 的基础知识,包括 DriverManager、Connection、Statement、ResultSet 等。 Filter 是指在 Servlet 中使用的过滤器,用于在 Servlet 请求和响应过程中执行某些操作。Filter 的基础...

    Java创建和关闭数据库连接的方法附代码.rar

    Java使用JDBC创建和关闭数据库连接的方法附代码,学习和熟悉一下JDBC的用法,创建一个数据库连接,并关闭或断开这个连接,面向java初学者。Class.forName("com.microsoft.jdbc.sqlserver.SQLServerDriver"); //加载...

Global site tag (gtag.js) - Google Analytics