笑看風云

          一切從頭開始
          posts - 28, comments - 1, trackbacks - 0, articles - 2

          Class.forName 介紹

          Posted on 2008-07-21 21:38 笑看風云 閱讀(144) 評論(0)  編輯  收藏 所屬分類: Java

          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()方法的時候,就必須保證:1、這個類已經加載;2、這個類已經連接了。而完成上面兩個步驟的正是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類的實例
          1、forName(String className)單參數時, initialize=true
              a.總是使用當前類裝載器(也就是裝載執行forName()請求的類  的類裝載器)
              b.總是初始化這個被裝載的類(當然也包括:裝載、連接、初始化)
          2、forName(String className, boolean initialize, ClassLoader loader)
              a.loader指定裝載參數類所用的類裝載器,如果null則用bootstrp裝載器。
              b.initialize=true時,肯定連接,而且初始化了;
              c.false時,絕對不會初始化,但是可能被連接了,但是這里有個例外,如果在調用這個forName()前,已經被初始化了,那么返回的類型也肯定是被初始化的(當然,這里也暗含著:被同一個loader所裝載的,而且這個類被初始化了)

          關于用戶自定義的類裝載器的loadClass()方法
          1、loadClass(String name)單參數時, resolve=false
              a.如果這個類已經被這個類裝載器所裝載,那么,返回這個已經被裝載的類型的Class的實例,否則,就用這個自定義的類裝載器來裝載這個class,這時不知道是否被連接。絕對不會被初始化
              b.這時唯一可以保證的是,這個類被裝載了。但是不知道這個類是不是被連接和初始化了
          2、loadClass(String name, boolean resolve)
              a.resolve=true時,則保證已經裝載,而且已經連接了。resolve=falses時,則僅僅是去裝載這個類,不關心是否連接了,所以此時可能被連接了,也可能沒有被連接


           原文地址 http://hi.baidu.com/zhengguobest/blog/item/13adc2fcdd422082b901a086.html
          只有注冊用戶登錄后才能發表評論。


          網站導航:
           
          主站蜘蛛池模板: 静宁县| 柳林县| 远安县| 叶城县| 融水| 灵台县| 泌阳县| 栾城县| 阜康市| 靖安县| 兰西县| 蒙城县| 醴陵市| 镇雄县| 黔西| 河间市| 东乡县| 荣成市| 平潭县| 文成县| 正镶白旗| 孝义市| 庐江县| 吉安县| 保定市| 阿勒泰市| 阿拉尔市| 石河子市| 织金县| 平和县| 阜宁县| 元谋县| 如东县| 南丹县| 南京市| 云浮市| 南郑县| 万荣县| 武陟县| 京山县| 澄迈县|