Class.forName 和 ClassLoader
在 java 中 Class.forName() 和 ClassLoader 都可以對類進(jìn)行加載。ClassLoader 就是遵循雙親委派模型最終調(diào)用啟動類加載器的類加載器,實(shí)現(xiàn)的功能是“通過一個類的全限定名來獲取描述此類的二進(jìn)制字節(jié)流”,獲取到二進(jìn)制流后放到 JVM 中。Class.forName() 方法實(shí)際上也是調(diào)用的 CLassLoader 來實(shí)現(xiàn)的。Class.forName()的源碼:
/*
* @param className the fully qualified name of the desired class.
* @return the {@code Class} object for the class with the
* specified name.
* @exception LinkageError if the linkage fails
* @exception ExceptionInInitializerError if the initialization provoked
* by this method fails
* @exception ClassNotFoundException if the class cannot be located
*/
@CallerSensitive
public static Class<?> forName(String className)
throws ClassNotFoundException {
Class<?> caller = Reflection.getCallerClass();
return forName0(className, true, ClassLoader.getClassLoader(caller), caller);
}
/** Called after security check for system loader access checks have been made. */
private static native Class<?> forName0(String name, boolean initialize,
ClassLoader loader,
Class<?> caller)
throws ClassNotFoundException;
forName(String className)會調(diào)用forName0方法,forName0方法是native方法。在這個 forName0 方法中的第二個參數(shù)被默認(rèn)設(shè)置為了 true,這個參數(shù)代表是否對加載的類進(jìn)行初始化,設(shè)置為 true 時會類進(jìn)行初始化,代表會執(zhí)行類中的靜態(tài)代碼塊,以及對靜態(tài)變量的賦值等操作。forName0方法的第三個參數(shù)是調(diào)用者的類加載器,通過這個加載器來進(jìn)行類的加載動作。
舉例
下面還是舉例來說明結(jié)果吧:一個含有靜態(tài)代碼塊、靜態(tài)變量、賦值給靜態(tài)變量的靜態(tài)方法的類。

測試方法:

運(yùn)行結(jié)果:

根據(jù)運(yùn)行結(jié)果得出 Class.forName 加載類是將類進(jìn)了初始化,而 ClassLoader 的 loadClass 并沒有對類進(jìn)行初始化,只是把類加載到了虛擬機(jī)中。
應(yīng)用場景
我們使用 JDBC 時通常是使用 Class.forName() 方法來加載數(shù)據(jù)庫連接驅(qū)動。這是因?yàn)樵?JDBC 規(guī)范中明確要求 Driver(數(shù)據(jù)庫驅(qū)動)類必須向 DriverManager 注冊自己。以 MySQL 的驅(qū)動為例解釋:

我們看到 Driver 注冊到 DriverManager 中的操作寫在了靜態(tài)代碼塊中,這就是為什么在寫 JDBC 時使用 Class.forName() 的原因了。