JDBC中为什么Class.forname("driverName")就可以创建驱动对象

  使用JDBC时,我们都会很自然得使用下列语句:

java 代码
  1. Class.forName("com.mysql.jdbc.Driver");   
  2. String url = "jdbc:mysql://127.0.0.1/test?useUnicode=true&characterEncoding=utf-8";   
  3. String user = "";   
  4. String psw = "";   
  5. Connection con = DriverManager.getConnection(url,user,psw);  


    为什么说很自然呢,因为无论是网上还是书本教程上得例子都是这样的,而且程序也确实正常运行了,于是大家也就心安理得的找葫芦画瓢下去了。
    一定要有这一句吗?不是的,我们完全可以用这样一句代替它:

java 代码
  1. com.mysql.jdbc.Driver driver = new com.mysql.jdbc.Driver();   
  2. //or:   
  3. //new com.mysql.jdbc.Driver();   
  4. String url = "jdbc:mysql://127.0.0.1/test?useUnicode=true&characterEncoding=utf-8";   
  5. String user = "";   
  6. String psw = "";   
  7. Connection con = DriverManager.getConnection(url,user,psw);   
  8.   

    大家可能都看出个大概来了,我们只需要在调用DriverManager的getConnection方法之前,保证相应的Driver类已经被加载到 jvm中,并且完成了类的初始化工作就行了,而具体是怎样实现这个功能却是没有讲究的。上面两种方法都可以实现这个功能,因此程序可以正常运行。注意了, 如果我们进行如下操作,程序是不能正常运行的,因为这样仅仅使Driver类被装载到jvm中,却没有进行相应的初始化工作。

java 代码
  1. com.mysql.jdbc.Driver driver = null;   
  2. //or:   
  3. ClassLoader cl = new ClassLoader();   
  4. cl.loadClass("com.mysql.jdbc.Driver");  


     我们都知道JDBC是使用Bridge模式进行设计的,DriverManager就是其中的Abstraction,java.sql.Driver是 Implementor,com.mysql.jdbc.Driver是Implementor的一个具体实现(请参考GOF的Bridge模式的描 述)。大家注意了,前一个Driver是一个接口,后者却是一个类,它实现了前面的Driver接口。
     Bridge模式中,Abstraction(DriverManager)是要拥有一个Implementor(Driver)的引用的,但是我们在使 用过程中,并没有将Driver对象注册到DriverManager中去啊,这是怎么回事呢?jdk文档对Driver的描述中有这么一句:
     When a Driver class is loaded, it should create an instance of itself and register it with the DriverManager
哦,原来是com.mysql.jdbc.Driver在装载完后自动帮我们完成了这一步骤。源代码是这样的:

java 代码
  1. package com.mysql.jdbc   
  2.   
  3. public class Driver extends NonRegisteringDriver implements java.sql.Driver {   
  4.  // ~ Static fields/initializers   
  5.  // --------------------------------------------- //   
  6.  // Register ourselves with the DriverManager   
  7.  //   
  8.  static {   
  9.     t ry {   
  10.               java.sql.DriverManager.registerDriver(new Driver());   
  11.           } catch (SQLException E) {   
  12.               throw new RuntimeException("Can't register driver!");   
  13.           }   
  14.   }   
  15. // ~ Constructors   
  16.  // -----------------------------------------------------------   
  17. /**  
  18.   * Construct a new driver and register it with DriverManager  
  19.   *   
  20.   * @throws SQLException  
  21.   *             if a database error occurs.  
  22.   */  
  23.  public Driver() throws SQLException {   
  24.      // Required for Class.forName().newInstance()   
  25.  }   

 

××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××

××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××

 

 

 

 

   开始接触JDBC时,一直有一个疑虑,为什么执行 Class.forName(“com.mysql.jdbc.Driver“)就可以载入MySql的驱动程序?JDBC的驱动程序初始化过程是怎么样 的?连接具体的数据库时,JDBC的DriverManager又是如何运作的?带着这么几个疑惑,本人下载了MySql的驱动源代码,结合J2SDK的 源代码,分析了一下JDBC的驱动管理机制。
    
1.     分析JDBC的驱动程序管理部分的实现代码:
    
        在 JDBC的层次上,sun主要定义了1个接口Driver和两个类:DirverManager和DriverInfo。每个JDBC驱动程序必须实现 Driver接口(在MySql的Connector/J驱动中,这个叫做com.mysql.jdbc.Driver)。而DriverManager 则负责管理所有的Driver对象,包含注册Driver;选择合适的Driver来建立到某个数据库的连接;以及进行一些Driver的信息管理等。 DriverInfo非常简单,用于保存Driver的信息,只有3个成员变量,Driver,DriverClass和 DriverClassName,意义非常明显。

        先看一下在DriverManager.java中的关键代码:
        private static java.util.Vector drivers = new java.util.Vector();
        所有的Driver对象保存在一个Vector数组中。

        注册Driver的函数叫registerDriver,将需要注册的Driver对象传入即可:
        public static synchronized void registerDriver(java.sql.Driver driver)
         throws SQLException {
             if (!initialized) {                                    //如果没有初始化,则先初始化
                 initialize();
             }      
             DriverInfo di = new DriverInfo();         //实际保存的不是Driver,而是一个DriverInfo对象,但是DriverInfo的其它成员 完全可以由Driver推导出来,所以个人觉得DriverInfo对象可有可无,直接使用Driver应该就可以了。
             di.driver = driver;
             di.driverClass = driver.getClass();
             di.driverClassName = di.driverClass.getName();
             drivers.addElement(di);                        //将DriverInfo对象添加到数组中
             println("registerDriver: " + di);
        }

        这样就完成了驱动程序的注册过程。然后重点看一下建立数据库连接的代码,在getConnection函数中,省略了一些非关键代码:
            private static synchronized Connection getConnection(
 String url, java.util.Properties info, ClassLoader callerCL) throws SQLException {

         SQLException reason = null;
         //轮询所有的DriverInfo对象。
         for (int i = 0; i < drivers.size(); i++) {
             DriverInfo di = (DriverInfo)drivers.elementAt(i);
             try {
                  //使用DriverInfo中的Driver对象去做实际的连接数据库的工作
                  Connection result = di.driver.connect(url, info);
                  if (result != null) {
                      // Success!
                      println("getConnection returning " + di);
                      return (result);    //一旦成功连接,直接返回Connection对象,然后推出
                  }
                 } catch (SQLException ex) {
                        ...
                  }
             }        
             //这就是经常看到的出错信息--找不到合适的驱动程序。
             println("getConnection: no suitable driver");    
             throw new SQLException("No suitable driver", "08001");
        }
        由 上面的getConnection函数可以看到,真正实现数据库连接的是Driver对象的connect函数。而且可以看到,由于 DriverManager.getConnection使用的是一种轮询的方式,注册的驱动程序越多,连接速度会越慢。JDBC连接数据库的速度很慢, 是不是和这种实现方式有关联呢?怀着这个问题,本人下载了MySql的Connector/J驱动包,开始分析其connect函数的实现。

2.    分析MySql的注册和建立连接部分的代码:
        打开MySql的源码包,首先分析其Driver类的实现。发现Driver类的实现非常简单,
        public class Driver extends NonRegisteringDriver implements java.sql.Driver {
        static {
                try {
                    java.sql.DriverManager.registerDriver(new Driver());
                    } catch (SQLException E) {
                        throw new RuntimeException("Can't register driver!");
                }
            }                public Driver() throws SQLException {
                // Required for Class.forName().newInstance()
        }            
        可 以看到,有一段static代码,调用了DriverManager的registerDriver方法。这其实就解释了 Class.forName(“com.mysql.jdbc.Driver”)能够完成MySql驱动注册的问题。因为forName会导致这段 static代码被调用,从而间接调用了registerDriver,完成注册过程。
        
        com.mysql.jdbc.Driver 从com.mysql.jdbc.NonRegisteringDriver继承而来,实际上是NonReisteringDriver完成了 java.sql.Driver接口的实现工作。转移目标,分析NonRegisteringDriver的connect函数。

        NonRegisteringDriver.connect的实现也比较简单,正合我意:

        public java.sql.Connection connect(String url, Properties info)
        throws SQLException {

        //1.    分析传入的连接字符串.
        if ((props = parseURL(url, info)) == null) {        
            return null;
        }
        try {
            //2.     建立一个Connection对象完成实际的数据库连接工作
            Connection newConn = new com.mysql.jdbc.Connection(host(props),
                    port(props), props, database(props), url, this);
            return newConn;
        非 常简单,先parseURL,然后使用Connection去建立连接。parseURL只是简单的字符串分析,主要是分析传入的连接字符串是否满足 “jdbc:mysql://host:port/database“的格式,如果不满足,直接返回null,然后由DriverManager去试验下 一个Driver。如果满足,则建立一个Connection对象建立实际的数据库连接,这不是本人关注的问题,源码分析就此打住。

        这也就解释了第二个问题,DriverManager的轮询查询注册的Driver对象的工作方式所带来的性能代价并不是很大,主工作量只是parseURL函数。
        
        以上是本人分析JDBC的驱动管理机制和建立数据库连接的一个一段分析笔记,本人是java新手,如果出现错误或者遗漏,欢迎指正,也欢迎鄙视,^_^

 

 

 

 

 ××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××

×××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××××

 

 

 

 

Class.forName方法介绍

在java.lang.Class中,有两个重载的forName方法,分别是:

  1. static Class<?>forName(String className),该方法等价于Class.forName(className, true, this.getClass().getClassLoader())
  2. static Class<?>forName(String className, boolean initialize,ClassLoader loader),其中3个参数分别表示:className - 所需类的完全限定名,initialize - 是否必须初始化类,loader - 用于加载类的类加载器。

forName方法的作用就是:
使用给定的类加载器,返回与带有给定字符串名的类或接口相关联的 Class 对象。给定一个类或接口的完全限定名,此方法会试图定位、加载和链接该类或接口。指定的类加载器用于加载该类或接口,如果参数loader 为 null,则该类通过引导类加载器加载。只有 initialize 参数为 true且以前未被初始化时,才初始化该类。

其他的都很容易懂,就是当第二个参数为true时,到底初始化的是什么呢?可以通过下面这个例子来了解:

[java] view plain copy
  1. package wen.hui.test.forname;  
  2.   
  3. /** 
  4.  * @author whwang 
  5.  * 
  6.  */  
  7. public class TestClassForName {  
  8.   
  9.     public static void main(String[] args) throws ClassNotFoundException {  
  10.         (new TestClassForName()).loadClass();  
  11.     }  
  12.       
  13.     @SuppressWarnings("unchecked")  
  14.     public Class<Test> loadClass() throws ClassNotFoundException {  
  15. //      Class<Test> clazz = (Class<Test>) Class.forName("wen.hui.test.forname.Test");  
  16.         Class<Test> clazz = (Class<Test>) Class.forName("wen.hui.test.forname.Test"true, getClass().getClassLoader());  
  17.         return clazz;  
  18.     }  
  19.       
  20. }  
  21.   
  22. class Test {  
  23.       
  24.     static {  
  25.         System.err.println("类的静态初始化块");  
  26.     }  
  27.       
  28.     public Test() {  
  29.         System.err.println("实例化类");  
  30.     }  
  31.      

 运行后,打印:类的静态初始化块

没错,Class.forName的第二个参数为true时,就是要求JVM在加载类后,初始化类的静态字段和静态块。

JDBC中使用Class.forName("xxx")的意义

在Java开发特别是数据库开发中,经常会用到Class.forName( )这个方法。通过上面的介绍,已经了解了Class.forName()方法的作用就是为了动态加载类,并决定是否需要初始化类的静态部分,而在JDBC 规范中明确要求Driver(数据库驱动)类必须向DriverManager注册自己。写到这里,相信大家都应该明白为什么在我们加载数据库驱动包的时 候有的仅仅需要Class.forName(xxx);而有的需要Class.forName(xxx).newInstance()。

下面以MySQL为例子,来看看MySQL的com.mysql.jdbc.Driver类是怎么写的,MySQL的Driver类源码(5.x)

[java] view plain copy
  1. public class Driver extends NonRegisteringDriver implements java.sql.Driver {  
  2.     // ~ Static fields/initializers  
  3.     // ---------------------------------------------  
  4.   
  5.     //  
  6.     // Register ourselves with the DriverManager  
  7.     //  
  8.     static {  
  9.         try {  
  10.             java.sql.DriverManager.registerDriver(new Driver());  
  11.         } catch (SQLException E) {  
  12.             throw new RuntimeException("Can't register driver!");  
  13.         }  
  14.     }  
  15.   
  16.     // ~ Constructors  
  17.     // -----------------------------------------------------------  
  18.   
  19.     /** 
  20.      * Construct a new driver and register it with DriverManager 
  21.      *  
  22.      * @throws SQLException 
  23.      *             if a database error occurs. 
  24.      */  
  25.     public Driver() throws SQLException {  
  26.         // Required for Class.forName().newInstance()  
  27.     }  
  28. }  

在使用JDBC连接MySQL数据库时,使用Class.forName("com.mysql.jdbc.Driver")就是为了向 DriverManager注册自己;当然使用 Class.forName("com.mysql.jdbc.Driver").newInstance()当然也没错,只是没有必要,因为后者还会生 成Driver类的实例,而这个是我们没有用的,没有必要创建它。如果在Driver类中那个static块里面的部分写在了构造方法中,那么就必须使用 Class.forName("com.mysql.jdbc.Driver").newInstance()来向DriverManager注册了。

 

 

 

 

 

 

 

 

 

 

 

 

 

 

你可能感兴趣的:(Class.forName)