jdbc中class.forname的作用
使用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是两码事,一个实例化类,一个加载类
上一篇: c#文件名/路径处理方法示例
下一篇: 2016年iOS公开可利用漏洞总结