Class.forName() 和 ClassLoader.loadClass()的區別? java
Class.forName() 和 Class.forName().NewInstance()的區別?mysql
Class.forName("xx.xx")等同於Class.forName("xx.xx",true,CALLClass.class.getClassLoader()),第二個參數(boolean)表示裝載類的時候是否初始化該類,即調用類的靜態塊的語句及初始化靜態成員變量。sql
ClassLoader loader = Thread.currentThread.getContextClassLoader(); //也能夠用(ClassLoader.getSystemClassLoader())編程
Class cls = loader.loadClass("xx.xx"); //這句話沒有執行初始化,其實與Class.forName("xx.xx",false,loader)是一致的,只是loader.loadClass("xx.xx")執行的是更底層的操做。this
只有執行cls.NewInstance()纔可以初始化類,獲得該類的一個實例spa
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。ci
區別就出來了。Class.forName(className)裝載的class已經被初始化,而ClassLoader.loadClass(className)裝載的class尚未被link。
通常狀況下,這兩個方法效果同樣,都能裝載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)。get