jdbc中class.forname的作用
使用JDBC時(shí),我們都會(huì)很自然得使用下列語(yǔ)句:
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);
為什么說(shuō)很自然呢,因?yàn)闊o(wú)論是網(wǎng)上還是書(shū)本教程上得例子都是這樣的,而且程序也確實(shí)正常運(yùn)行了,于是大家也就心安理得的找葫蘆畫(huà)瓢下去了。
一定要有這一句嗎?不是的,我們完全可以用這樣一句代替它:
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);
大家可能都看出個(gè)大概來(lái)了,我們只需要在調(diào)用DriverManager的getConnection方法之前,保證相應(yīng)的Driver類已經(jīng)被加載到j(luò)vm中,并且完成了類的初始化工作就行了,而具體是怎樣實(shí)現(xiàn)這個(gè)功能卻是沒(méi)有講究的。上面兩種方法都可以實(shí)現(xiàn)這個(gè)功能,因此程序可以正常運(yùn)行。注意了,如果我們進(jìn)行如下操作,程序是不能正常運(yùn)行的,因?yàn)檫@樣僅僅使Driver類被裝載到j(luò)vm中,卻沒(méi)有進(jìn)行相應(yīng)的初始化工作。
com.mysql.jdbc.Driver driver = null;
//or:
ClassLoader cl = new ClassLoader();
cl.loadClass("com.mysql.jdbc.Driver");
我們都知道JDBC是使用Bridge模式進(jìn)行設(shè)計(jì)的,DriverManager就是其中的Abstraction,java.sql.Driver是Implementor,com.mysql.jdbc.Driver是Implementor的一個(gè)具體實(shí)現(xiàn)(請(qǐng)參考GOF的Bridge模式的描述)。大家注意了,前一個(gè)Driver是一個(gè)接口,后者卻是一個(gè)類,它實(shí)現(xiàn)了前面的Driver接口。
Bridge模式中,Abstraction(DriverManager)是要擁有一個(gè)Implementor(Driver)的引用的,但是我們?cè)谑褂眠^(guò)程中,并沒(méi)有將Driver對(duì)象注冊(cè)到DriverManager中去啊,這是怎么回事呢?jdk文檔對(duì)Driver的描述中有這么一句:
When a Driver class is loaded, it should create an instance of itself and register it with the DriverManager
哦,原來(lái)是com.mysql.jdbc.Driver在裝載完后自動(dòng)幫我們完成了這一步驟。源代碼是這樣的:
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驅(qū)動(dòng)的裝載
ClassLoader cl = Thread.currentThread().getContextClassLoader();
Class clazz = cl.loadClass("com.mysql.jdbc.Driver");
clazz.newInstance();
Connection conn = DriverManager.getConnection("jdbcurl");
同樣可以執(zhí)行。但是這樣就多構(gòu)造了一個(gè)com.mysql.jdbc.Driver實(shí)例。同Class.forName("com.mysql.jdbc.Driver")。
即:
Class.forName("com.mysql.jdbc.Driver")==cl.loadClass("com.mysql.jdbc.Driver").newInstance();
Class.forName和 ClassLoader.loadClass是兩碼事,一個(gè)實(shí)例化類,一個(gè)加載類
相關(guān)文章
Spring之spring-context-indexer依賴詳解
這篇文章主要介紹了Spring之spring-context-indexer依賴詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-11-11SpringBoot搭配AOP實(shí)現(xiàn)自定義注解
這篇文章主要為大家詳細(xì)介紹了SpringBoot如何搭配AOP實(shí)現(xiàn)自定義注解,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2022-12-12SpringBoot中的RestTemplate使用方法詳解
這篇文章主要介紹了SpringBoot中的RestTemplate使用方法詳解,為了方便使用,這里我封裝成一個(gè)工具類來(lái)靜態(tài)調(diào)用RestTemplate,基于SpringBoot2.4.2版本,需要的朋友可以參考下2024-01-01Spring Data JPA中findOne()和getOne()用法
這篇文章主要介紹了Spring Data JPA中findOne()和getOne()的用法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-11-11java.util.NoSuchElementException原因及兩種解決方法
本文主要介紹了java.util.NoSuchElementException原因及兩種解決方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-06-06IDEA2022.1創(chuàng)建maven項(xiàng)目規(guī)避idea2022新建maven項(xiàng)目卡死無(wú)反應(yīng)問(wèn)題
這篇文章主要介紹了IDEA2022.1創(chuàng)建maven項(xiàng)目規(guī)避idea2022新建maven項(xiàng)目卡死無(wú)反應(yīng)問(wèn)題,本文通過(guò)圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-08-08