欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

jdbc中class.forname的作用

程序员文章站 2024-02-18 20:42:59
使用jdbc时,我们都会很自然得使用下列语句:复制代码 代码如下:class.forname("com.mysql.jdbc.driver");   s...

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

复制代码 代码如下:

class.forname("com.mysql.jdbc.driver");  
string url = "jdbc:mysql://127.0.0.1/test?useunicode=true&characterencoding=utf-8";  
string user = "";  
string psw = "";  
connection con = drivermanager.getconnection(url,user,psw); 

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

复制代码 代码如下:

com.mysql.jdbc.driver driver = new com.mysql.jdbc.driver();  
//or:  
//new com.mysql.jdbc.driver(); 
string url = "jdbc:mysql://127.0.0.1/test?useunicode=true&characterencoding=utf-8";  
string user = "";  
string psw = "";  
connection con = drivermanager.getconnection(url,user,psw);

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

复制代码 代码如下:

com.mysql.jdbc.driver driver = null;  
//or:  
classloader cl = new classloader();  
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在装载完后自动帮我们完成了这一步骤。源代码是这样的:

复制代码 代码如下:

package com.mysql.jdbc  

public class driver extends nonregisteringdriver implements java.sql.driver {  
 // ~ static fields/initializers 
 // register ourselves with the drivermanager 
 // 
 static {  
    t ry {  
              java.sql.drivermanager.registerdriver(new driver());  
          } catch (sqlexception e) {  
              throw new runtimeexception("can't register driver!");  
          }  
  }  
// ~ constructors 
/** 
  * construct a new driver and register it with drivermanager
  *  
  * @throws sqlexception
  *             if a database error occurs.
  */ 
 public driver() throws sqlexception {  
     // required for class.forname().newinstance() 
 }  
}

ps:改修jdbc驱动的装载

复制代码 代码如下:

classloader cl = thread.currentthread().getcontextclassloader();
class clazz = cl.loadclass("com.mysql.jdbc.driver");
clazz.newinstance();
connection conn = drivermanager.getconnection("jdbcurl");

同样可以执行。但是这样就多构造了一个com.mysql.jdbc.driver实例。同class.forname("com.mysql.jdbc.driver")。

即:

复制代码 代码如下:

class.forname("com.mysql.jdbc.driver")==cl.loadclass("com.mysql.jdbc.driver").newinstance();

class.forname和 classloader.loadclass是两码事,一个实例化类,一个加载类