Class.forName(String name)java
接上一篇JDBC。原本这个内容是放在前面的一篇里面的一块儿的,后来发现越写越多,想一想看就算了,仍是单独开一篇文章好了,这样也能写得更加详细点。mysql
上一篇文章的第4点,getConnection()方法里面,我把从.properties里面获取mysqlpackage的地方替换成实际的value值,那么替换后的应该是Class.forName("com.mysql.jdbc.Driver"),实际上全部的JDBC链接先写的基本上也都是这一句。另外,哪怕删除这一句,也是能够运行成功的。那为何还要Class.forName("com.mysql.jdbc.Driver")呢?OK,分点讲解。算法
为何要Class.forName("com.mysql.jdbc.Driver")?sql
JDBC是23种模式中的桥接模式的典型应用,熟悉桥接模式的基本上稍微看一下源代码就知道为何了,那桥接模式这里不讲,只讲为何要这么作。Class.forName(String className)的做用有两个,第一是CLASSPATH下指定名字的.class文件加载到Java虚拟机内存中, 第二是初始化这个类。看到这句话,返回值都没有,那写在这里的做用很明显了,就是初始化"com.mysql.jdbc.Drvier"。初始化作了什么?给静态资源赋值以及执行静态代码块,因此,反编译一下"mysql-connector-java-5.1.20-bin.jar"这个jar包,查看一下Driver类:数据库
public class Driver extends NonRegisteringDriver
implements java.sql.Driver { public Driver() throws SQLException { } static { try { DriverManager.registerDriver(new Driver()); } catch (SQLException E) { throw new RuntimeException("Can't register driver!"); } } }
看到Class.forName("com.mysql.jdbc.Driver")的做用实际上就是调用DriverManager的registerDriver方法注册一个mysql的JDBC驱动(Driver)而已,Driver继承NonRegisteringDriver.java,NonRegisteringDriver.java实现了JDK提供的Driver接口,这个Driver提供了若干数据库链接的方法,每一个不一样的数据库链接类都必须实现它,并重写和具体的数据库链接的算法。DriverManager也是JDK中的类,截一些关键代码:数组
private static final CopyOnWriteArrayList<DriverInfo> registeredDrivers = new CopyOnWriteArrayList();
public static synchronized void registerDriver(Driver paramDriver) throws SQLException { if (paramDriver != null) registeredDrivers.addIfAbsent(new DriverInfo(paramDriver)); else throw new NullPointerException(); println("registerDriver: " + paramDriver); }
底层利用了一个CopyOnWriteArrayList做为容器(这是一个线程安全的容器,不过每次add的时候都会对底层数组进行一次新的复制,因此在读远多于写的时候建议可使用这个),放那些注册进去的DriverInfo。最终getConnection(...)的时候就拿registerDrivers里面注册进去的具体的某个数据库的DriverInfo(像MySql的Driver就在DriverInfo里面)去链接具体的数据库。OK,因此总结一下整个流程:安全
JDK不负责和数据库链接打交道,也不必,只提供一个具体的接口Driver,告诉全部第三方,要链接数据库,就去实现这个接口,而后经过DriverManager注册一下,到时候链接某个数据库的时候,你已经在我这里注册了,我会调用你注册进来的Driver里面的方法去对指定数据库进行链接的。而后Mysql就实现本身的Driver,Oracle就实现本身的Driver,经过static块注册一下,再而后,就没有而后了。spa
为何不直接new?线程
意思是这么写"com.mysql.jdbc.Driver d = new com.mysql.jdbc.Driver();",能够啊,由于在new的时候会自动触发对一个类的初始化。问题是new出来干吗?com.mysql.jdbc.Driver里面的方法咱们会用到吗,而且咱们知道怎么用吗?仅仅为了初始化一个类而new一个类实例出来还不如不去new,直接使用Class.forName(String name)初始化就能够了。DriverManager类的getConnection(...)方法的存在自己就是帮助用户调用Driver里面的各类方法链接数据库,JDK都作好了,开发者就不必本身写了。code
为何删除Class.forName("com.mysql.jdbc.Driver")仍是能够运行?
1996年1月23日JDK1.0发布,Java语言有了第一个正式版本的运行环境。JDBC是1997年2月19日,在JDK1.1的版本中发布的,从版本就看得出,JDBC属于Java技术的一些最基础的功能点。那在JDK1.5以后,其实已经不须要去显式调用Class.forName("com.mysql.jdbc.Driver")了,DriverManager会自动去加载合适的驱动,可是前提是CLASSPATH下必须有驱动jar包。