Class.forName 介紹

Class.forName(xxx.xx.xx) 返回的是一個類  首先你要明白在java裏面任何class都要裝載在虛擬機上才能運行。這句話就是裝載類用的(和new 不同,要分清楚)。  至於何時用,你能夠考慮一下這個問題,給你一個字符串變量,它表明一個類的包名和類名,你怎麼實例化它?只有你提到的這個方法了,不過要再加一點。  A a = (A)Class.forName("pacage.A").newInstance();  這和你  A a = new A();  是同樣的效果。  關於補充的問題  答案是確定的,jvm會執行靜態代碼段,你要記住一個概念,靜態代碼是和class綁定的,class裝載成功就表示執行了你的靜態代碼了。並且之後不會再走這段靜態代碼了。  Class.forName(xxx.xx.xx)返回的是一個類  Class.forName(xxx.xx.xx)的做用是要求JVM查找並加載指定的類,也就是說JVM會執行該類的靜態代碼段  動態加載和建立Class 對象,好比想根據用戶輸入的字符串來建立對象  String str = "用戶輸入的字符串" ;  Class t = Class.forName(str);  t.newInstance();  在初始化一個類,生成一個實例的時候,newInstance()方法和new關鍵字除了一個是方法,一個是關鍵字外,最主要有什麼區別?它們的區別在於建立對象的方式不同,前者是使用類加載機制,後者是建立一個新類。那麼爲何會有兩種建立對象方式?這主要考慮到軟件的可伸縮、可擴展和可重用等軟件設計思想。  Java中工廠模式常用newInstance()方法來建立對象,所以從爲何要使用工廠模式上能夠找到具體答案。 例如:  class c = Class.forName(「Example」);  factory = (ExampleInterface)c.newInstance();  其中ExampleInterface是Example的接口,能夠寫成以下形式:  String className = "Example";  class c = Class.forName(className);  factory = (ExampleInterface)c.newInstance();  進一步能夠寫成以下形式:  String className = readfromXMlConfig;//從xml 配置文件中得到字符串  class c = Class.forName(className);  factory = (ExampleInterface)c.newInstance();  上面代碼已經不存在Example的類名稱,它的優勢是,不管Example類怎麼變化,上述代碼不變,甚至能夠更換Example的兄弟類Example2 , Example3 , Example4……,只要他們繼承ExampleInterface就能夠。  從JVM的角度看,咱們使用關鍵字new建立一個類的時候,這個類能夠沒有被加載。可是使用newInstance()方法的時候,就必須保證:一、這個類已經加載;二、這個類已經鏈接了。而完成上面兩個步驟的正是Class的靜態方法forName()所完成的,這個靜態方法調用了啓動類加載器,即加載 java API的那個加載器。  如今能夠看出,newInstance()其實是把new這個方式分解爲兩步,即首先調用Class加載方法加載某個類,而後實例化。 這樣分步的好處是顯而易見的。咱們能夠在調用class的靜態加載方法forName時得到更好的靈活性,提供給了一種降耦的手段。  最後用最簡單的描述來區分new關鍵字和newInstance()方法的區別:  newInstance: 弱類型。低效率。只能調用無參構造。  new: 強類型。相對高效。能調用任何public構造。  下面內容轉自  http://blog.csdn.net/iceman1952/archive/2007/03/07/1523025.aspx  介紹的是 forName() 和 ClassLoader 的 loadClass 方法。  如今終於知道了爲何 forName()是會執行 static 語句,由於默認狀況它老是初始化這個被裝載的類。  關於forName()方法  這個方法老是返回要加載的類的Class類的實例  一、forName(String className)單參數時, initialize=true      a.老是使用當前類裝載器(也就是裝載執行forName()請求的類  的類裝載器)      b.老是初始化這個被裝載的類(固然也包括:裝載、鏈接、初始化)  二、forName(String className, boolean initialize, ClassLoader loader)      a.loader指定裝載參數類所用的類裝載器,若是null則用bootstrp裝載器。       b.initialize=true時,確定鏈接,並且初始化了;       c.false時,絕對不會初始化,可是可能被鏈接了,可是這裏有個例外,若是在調用這個forName()前,已經被初始化了,那麼返回的類型也確定是被初始化的(固然,這裏也暗含着:被同一個loader所裝載的,並且這個類被初始化了)  關於用戶自定義的類裝載器的loadClass()方法  一、loadClass(String name)單參數時, resolve=false      a.若是這個類已經被這個類裝載器所裝載,那麼,返回這個已經被裝載的類型的Class的實例,不然,就用這個自定義的類裝載器來裝載這個class,這時不知道是否被鏈接。絕對不會被初始化      b.這時惟一能夠保證的是,這個類被裝載了。可是不知道這個類是否是被鏈接和初始化了  二、loadClass(String name, boolean resolve)      a.resolve=true時,則保證已經裝載,並且已經鏈接了。resolve=falses時,則僅僅是去裝載這個類,不關心是否鏈接了,因此此時可能被鏈接了,也可能沒有被鏈接 
相關文章
相關標籤/搜索