使用jdbc时,我们都会很自然得使用下列语句:
Java代码
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);
一定要有这一句吗?不是的,我们完全可以用这样一句代替它:
Java代码
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中,却没有进行相应的初始化工作。
Java代码
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在装载完后自动帮我们完成了这一步骤。源代码是这样的:
java 代码
Java代码
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() } }
作者“tianlanlan”