天天看點

Java中Class.forName和ClassLoader.loadClass的差別

Class.forName("xx.xx")等同于Class.forName("xx.xx",true,CALLClass.class.getClassLoader()),第二個參數(bool)表示裝載類的時候是否初始化該類,即調用類的靜态塊的語句及初始化靜态成員變量。

ClassLoader loader = Thread.currentThread.getContextClassLoader(); //也可以用(ClassLoader.getSystemClassLoader())

Class cls = loader.loadClass("xx.xx"); //這句話沒有執行初始化,其實與Class.forName("xx.xx",false,loader)是一緻的,隻是loader.loadClass("xx.xx")執行的是更底層的操作。

隻有執行cls.NewInstance()才能夠初始化類,得到該類的一個執行個體

Class的裝載分了三個階段,loading,linking和initializing,分别定義在The Java Language Specification的12.2,12.3和12.4。
Class.forName(className) 實際上是調用Class.forName(className, true, this.getClass().getClassLoader())。注意第二個參數,是指Class被loading後是不是必須被初始化。
ClassLoader.loadClass(className)實際上調用的是ClassLoader.loadClass(name, false),第二個參數指出Class是否被link。
差別就出來了。Class.forName(className)裝載的class已經被初始化,而ClassLoader.loadClass(className)裝載的class還沒有被link。
forName支援數組類型,loadClass不支援數組
一般情況下,這兩個方法效果一樣,都能裝載Class。但如果程式依賴于Class是否被初始化,就必須用Class.forName(name)了。
例如,在JDBC程式設計中,常看到這樣的用法,Class.forName("com.mysql.jdbc.Driver"),如果換成了 getClass().getClassLoader().loadClass("com.mysql.jdbc.Driver"),就不行。
為什麼呢?打開com.mysql.jdbc.Driver的源代碼看看,
//
// Register ourselves with the DriverManager
//
static {
    try {
        java.sql.DriverManager.registerDriver(new Driver());
    } catch (SQLException E) {
        throw new RuntimeException("Can't register driver!");
    }
}
原來,Driver在static塊中會注冊自己到java.sql.DriverManager。而static塊就是在Class的初始化中被執行。是以這個地方就隻能用Class.forName(className)。