- 論壇徽章:
- 0
|
[color="#295200"]Class.forName 介紹
Class.forName(xxx.xx.xx) 返回的是一個類
首先你要明白在java里面任何class都要裝載在虛擬機(jī)上才能運(yùn)行。這句話就是裝載類用的(和new 不一樣,要分清楚)。
至于什么時候用,你可以考慮一下這個問題,給你一個字符串變量,它代表一個類的包名和類名,你怎么實(shí)例化它?只有你提到的這個方法了,不過要再加一點(diǎn)。
A a = (A)Class.forName("pacage.A").newInstance();
這和你
A a = new A();
是一樣的效果。
關(guān)于補(bǔ)充的問題
答案是肯定的,jvm會執(zhí)行靜態(tài)代碼段,你要記住一個概念,靜態(tài)代碼是和class綁定的,class裝載成功就表示執(zhí)行了你的靜態(tài)代碼了。而且以后不會再走這段靜態(tài)代碼了。
Class.forName(xxx.xx.xx) 返回的是一個類
Class.forName(xxx.xx.xx);的作用是要求JVM查找并加載指定的類,也就是說JVM會執(zhí)行該類的靜態(tài)代碼段
動態(tài)加載和創(chuàng)建Class 對象,比如想根據(jù)用戶輸入的字符串來創(chuàng)建對象
String str = 用戶輸入的字符串
Class t = Class.forName(str);
t.newInstance();
在初始化一個類,生成一個實(shí)例的時候,newInstance()方法和new關(guān)鍵字除了一個是方法,一個是關(guān)鍵字外,最主要有什么區(qū)別?它們的
區(qū)別在于創(chuàng)建對象的方式不一樣,前者是使用類加載機(jī)制,后者是創(chuàng)建一個新類。那么為什么會有兩種創(chuàng)建對象方式?這主要考慮到軟件的可伸縮、可擴(kuò)展和可重用
等軟件設(shè)計思想。
Java中工廠模式經(jīng)常使用newInstance()方法來創(chuàng)建對象,因此從為什么要使用工廠模式上可以找到具體答案。 例如:
class c = Class.forName(“Example”);
factory = (ExampleInterface)c.newInstance();
其中ExampleInterface是Example的接口,可以寫成如下形式:
String className = "Example";
class c = Class.forName(className);
factory = (ExampleInterface)c.newInstance();
進(jìn)一步可以寫成如下形式:
String className = readfromXMlConfig;//從xml 配置文件中獲得字符串
class c = Class.forName(className);
factory = (ExampleInterface)c.newInstance();
上面代碼已經(jīng)不存在Example的類名稱,它的優(yōu)點(diǎn)是,無論Example類怎么變化,上述代碼不變,甚至可以更換Example的兄弟類Example2 , Example3 , Example4……,只要他們繼承ExampleInterface就可以。
從JVM的角度看,我們使用關(guān)鍵字new創(chuàng)建一個類的時候,這個類可以沒有被加載。但是使用newInstance()方法的時候,就必須保證:1、這個
類已經(jīng)加載;2、這個類已經(jīng)連接了。而完成上面兩個步驟的正是Class的靜態(tài)方法forName()所完成的,這個靜態(tài)方法調(diào)用了啟動類加載器,即加載
java API的那個加載器。
現(xiàn)在可以看出,newInstance()實(shí)際上是把new這個方式分解為兩步,即首先調(diào)用Class加載方法加載某個類,然后實(shí)例化。 這樣分步的好處是顯而易見的。我們可以在調(diào)用class的靜態(tài)加載方法forName時獲得更好的靈活性,提供給了一種降耦的手段。
最后用最簡單的描述來區(qū)分new關(guān)鍵字和newInstance()方法的區(qū)別:
newInstance: 弱類型。低效率。只能調(diào)用無參構(gòu)造。
new: 強(qiáng)類型。相對高效。能調(diào)用任何public構(gòu)造。
下面內(nèi)容轉(zhuǎn)自
http://blog.csdn.net/iceman1952/archive/2007/03/07/1523025.aspx
介紹的是 forName() 和 ClassLoader 的 loadClass 方法。
現(xiàn)在終于知道了為什么 forName()是會執(zhí)行 static 語句,因為默認(rèn)情況它總是初始化這個被裝載的類。
關(guān)于forName()方法
這個方法總是返回要加載的類的Class類的實(shí)例
1、forName(String className)單參數(shù)時, initialize=true
a.總是使用當(dāng)前類裝載器(也就是裝載執(zhí)行forName()請求的類 的類裝載器)
b.總是初始化這個被裝載的類(當(dāng)然也包括:裝載、連接、初始化)
2、forName(String className, boolean initialize, ClassLoader loader)
a.loader指定裝載參數(shù)類所用的類裝載器,如果null則用bootstrp裝載器。
b.initialize=true時,肯定連接,而且初始化了;
c.false時,絕對不會初始化,但是可能被連接了,但是這里有個例外,如果在調(diào)用這個forName()前,已經(jīng)被初始化了,那么返回的類型也肯定是被初始化的(當(dāng)然,這里也暗含著:被同一個loader所裝載的,而且這個類被初始化了)
關(guān)于用戶自定義的類裝載器的loadClass()方法
1、loadClass(String name)單參數(shù)時, resolve=false
a.如果這個類已經(jīng)被這個類裝載器所裝載,那么,返回這個已經(jīng)被裝載的類型的Class的實(shí)例,否則,就用這個自定義的類裝載器來裝載這個class,這時不知道是否被連接。絕對不會被初始化
b.這時唯一可以保證的是,這個類被裝載了。但是不知道這個類是不是被連接和初始化了
2、loadClass(String name, boolean resolve)
a.resolve=true時,則保證已經(jīng)裝載,而且已經(jīng)連接了。resolve=falses時,則僅僅是去裝載這個類,不關(guān)心是否連接了,所以此時可能被連接了,也可能沒有被連接
[color="#000099"]原文地址
http://hi.baidu.com/zhengguobest/blog/item/13adc2fcdd422082b901a086.html
本文來自ChinaUnix博客,如果查看原文請點(diǎn):http://blog.chinaunix.net/u3/111264/showart_2162054.html |
|