public class ClassForNameClassLoaderTest {
public static String nihao = "nihao";
static{
System.out.println ("我是靜態(tài)代碼塊");
}
{
System.out.println ("我是非靜態(tài)代碼塊");
}
}
public class ClassLoaderTest {
public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException {
//執(zhí)行靜態(tài)代碼塊和靜態(tài)變量的賦值
Class<ClassForNameClassLoaderTest> aClass = ( Class<ClassForNameClassLoaderTest> ) Class.forName ("builder.link.ClassForNameClassLoaderTest");
//不執(zhí)行靜態(tài)代碼塊,靜態(tài)變量不會賦值
// Class<?> aClass = ClassLoader.getSystemClassLoader ().loadClass ("builder.link.ClassForNameClassLoaderTest");
//Class.forName再執(zhí)行下段代碼是,執(zhí)行了ClassForNameClassLoaderTest 的非靜態(tài)代碼塊
//如果是classloader執(zhí)行靜態(tài)代碼塊 非靜態(tài)代碼塊 靜態(tài)變量賦值
aClass.newInstance ();
}
}
- clssLoader和classForName都是用來加載類的径缅,但是這兩個方法一般卻又在不同的場景使用。classLoader一般是spring容器用來加載bean的時候使用的某弦,而classForName一般我們都是在使用數據庫驅動的時候會使用該方法。
- 了解了兩個加載類的不同場景之外后,先看一下類加載的過程治筒。
加載:class文件首先被加載到jvm中
鏈接:3個步驟酱床。
首先驗證:驗證是為了保證我們的.class文件的合法性羊赵。會分別盡心文件格式驗證,元數據驗證扇谣,字節(jié)碼驗證昧捷,符號引用驗證
然后準備:初始化靜態(tài)成員變量,并賦值
最后解析:將符號引用轉換為直接引用
初始化罐寨,執(zhí)行靜態(tài)代碼快靡挥。類裝載過程到此結束
classForname和classloader的底層實現
- classForname
public static Class<?> forName(String className)
throws ClassNotFoundException {
return forName0(className, true, ClassLoader.getCallerClassLoader());
}
/** Called after security checks have been made. */
private static native Class forName0(String name, boolean initialize,ClassLoader loader)
throws ClassNotFoundException;
這里第二個參數initaline表示是否對加載的類進行初始化。而我們的classforname傳入的是true鸯绿。所以classforname會對類進行初始化操作.
- classloader
原代碼中return loadClass(name, false);傳入的是false,所以loaderClass則對類不進行初始化跋破。
new和newInstance()的區(qū)別
從JVM的角度看簸淀,我們使用關鍵字new創(chuàng)建一個類的時候,這個類可以沒有被加載毒返。但是使用newInstance()方法的時候租幕,就必須保證:
1、這個類已經加載拧簸;
2劲绪、這個類已經連接了。
而完成上面兩個步驟的正是Class的靜態(tài)方法forName()所完成的狡恬,這個靜態(tài)方法調用了啟動類加載器珠叔,即加載 java API的那個加載器。
現在可以看出弟劲,newInstance()實際上是把new這個方式分解為兩步祷安,即首先調用Class加載方法加載某個類,然后實例化兔乞。 這樣分步的好處是顯而易見的汇鞭。我們可以在調用class的靜態(tài)加載方法forName時獲得更好的靈活性,提供給了一種降耦的手段庸追。
二.new 和Class.forName()有什么區(qū)別霍骄?
其實上面已經說到一些了,這里來做個總結:
首先淡溯,newInstance( )是一個方法读整,而new是一個關鍵字;
其次咱娶,Class下的newInstance()的使用有局限米间,因為它生成對象只能調用無參的構造函數,而使用 new關鍵字生成對象沒有這個限制膘侮。
簡言之:
newInstance(): 弱類型,低效率,只能調用無參構造屈糊。
new: 強類型,相對高效,能調用任何public構造。
Class.forName(“”)返回的是類琼了。
Class.forName(“”).newInstance()返回的是object 逻锐。
三.為什么在加載數據庫驅動包的時候有用的是Class.forName( ),卻沒有調用newInstance( )雕薪?
在Java開發(fā)特別是數據庫開發(fā)中昧诱,經常會用到Class.forName( )這個方法。
通過查詢Java Documentation我們會發(fā)現使用Class.forName( )靜態(tài)方法的目的是為了動態(tài)加載類所袁。
通常編碼過程中盏档,在加載完成后,一般還要調用Class下的newInstance( )靜態(tài)方法來實例化對象以便操作纲熏。因此妆丘,單單使用Class.forName( )是動態(tài)加載類是沒有用的,其最終目的是為了實例化對象局劲。
有數據庫開發(fā)經驗朋友會發(fā)現勺拣,為什么在我們加載數據庫驅動包的時候有的卻沒有調用newInstance( )方法呢?
即有的jdbc連接數據庫的寫法里是Class.forName(xxx.xx.xx);而有一 些:Class.forName(xxx.xx.xx).newInstance()鱼填,為什么會有這兩種寫法呢药有?
剛才提到,Class.forName(“”);的作用是要求JVM查找并加載指定的類苹丸,首先要明白愤惰,java里面任何class都要裝載在虛擬機上才能運行,而靜態(tài)代碼是和class綁定的赘理,class裝載成功就表示執(zhí)行了你的靜態(tài)代碼了宦言,而且以后不會再走這段靜態(tài)代碼了。
而我們前面也說了商模,Class.forName(xxx.xx.xx)的作用就是要求JVM查找并加載指定的類奠旺,如果在類中有靜態(tài)初始化器的話,JVM必然會執(zhí)行該類的靜態(tài)代碼段施流。
而在JDBC規(guī)范中明確要求這個Driver類必須向DriverManager注冊自己响疚,即任何一個JDBC Driver的 Driver類的代碼都必須類似如下:
public class MyJDBCDriver implements Driver {
static {
DriverManager.registerDriver(new MyJDBCDriver());
}
}
既然在靜態(tài)初始化器的中已經進行了注冊,所以我們在使用JDBC時只需要Class.forName(XXX.XXX);就可以了瞪醋。