1. 程式人生 > >Class.forName()方法詳解

Class.forName()方法詳解

1、Class類簡介:

 Java程式在執行時,Java執行時系統一直對所有的物件進行所謂的執行時型別標識。這項資訊紀錄了每個物件所屬的類。虛擬機器通常使用執行時型別資訊選準正確方法去執行,用來儲存這些型別資訊的類是Class類。Class類封裝一個物件和介面執行時的狀態,當裝載類時,Class型別的物件自動建立。
      Class 沒有公共構造方法。Class 物件是在載入類時由Java 虛擬機器以及通過呼叫類載入器中的 defineClass 方法自動構造的,因此不能顯式地宣告一個Class物件。 
      虛擬機器為每種型別管理一個獨一無二的Class物件。也就是說,每個類(型)都有一個Class物件。執行程式時,Java虛擬機器(JVM)首先檢查是否所要載入的類對應的Class物件是否已經載入。如果沒有載入,JVM就會根據類名查詢.class檔案,並將其Class物件載入。
      基本的 Java 型別(boolean、byte、char、short、int、long、float 和 double)和關鍵字 void 也都對應一個 Class 物件。 
      每個陣列屬於被對映為 Class 物件的一個類,所有具有相同元素型別和維數的陣列都共享該 Class 物件。
      一般某個類的Class物件被載入記憶體,它就用來建立這個類的所有物件。

一、如何得到Class的物件呢?有三種方法可以的獲取:

  1、呼叫Object類的getClass()方法來得到Class物件,這也是最常見的產生Class物件的方法。例如:
    MyObject x;
    Class c1 = x.getClass();

 2、使用Class類的中靜態forName()方法獲得與字串對應的Class物件。例如: 
    Class c2=Class.forName("MyObject"),Employee必須是介面或者類的名字。

3、獲取Class型別物件的第三個方法非常簡單。如果T是一個Java型別,那麼T.class就代表了匹配的類物件。例如
    Class cl1 = Manager.class;
    Class cl2 = int.class;
    Class cl3 = Double[].class;
    注意:Class物件實際上描述的只是型別,而這型別未必是類或者介面。例如上面的int.class是一個Class型別的物件。由於歷史原因,陣列型別的getName方法會返回奇怪的名字。

二、Class類的常用方法

1、getName() 

一個Class物件描述了一個特定類的屬性,Class類中最常用的方法getName以 String 的形式返回此 Class 物件所表示的實體(類、介面、陣列類、基本型別或 void)名稱。

2、newInstance()

Class還有一個有用的方法可以為類建立一個例項,這個方法叫做newInstance()。例如:
    x.getClass.newInstance(),建立了一個同x一樣型別的新例項。newInstance()方法呼叫預設構造器(無引數構造器)初始化新建物件。

3、getClassLoader() 

返回該類的類載入器。

   4、getComponentType() 
    返回表示陣列元件型別的 Class。

   5、getSuperclass() 
    返回表示此 Class 所表示的實體(類、介面、基本型別或 void)的超類的 Class。

   6、isArray() 
    判定此 Class 物件是否表示一個數組類。

三、Class的一些使用技巧

1、forName和newInstance結合起來使用,可以根據儲存在字串中的類名建立物件。例如
    Object obj = Class.forName(s).newInstance();

2、虛擬機器為每種型別管理一個獨一無二的Class物件。因此可以使用==操作符來比較類物件。例如:
    if(e.getClass() == Employee.class)...

2、 Class.forName()方法:

Class.forName返回與給定的字串名稱相關聯介面的Class物件。

Class.forName是一個靜態方法,同樣可以用來載入類。該方法有兩種形式:Class.forName(String name, boolean initialize, ClassLoader loader)和 Class.forName(String className)。第一種形式的引數 name表示的是類的全名;initialize表示是否初始化類;loader表示載入時使用的類載入器。第二種形式則相當於設定了引數 initialize的值為 true,loader的值為當前類的類載入器。

staticClass<?>

forName(StringclassName)

Returns the Classobject associated with the class or interface with the given string name.

staticClass<?>

forName(Stringname, boolean initialize,ClassLoaderloader)

Returns the Classobject associated with the class or interface with the given string name, using the given class loader.

說明:

publicstatic Class<?> forName(StringclassName)

Returns theClassobject associated withthe class or interface with the given string name. Invokingthis method is equivalent to:

Class.forName(className,true, currentLoader)

wherecurrentLoaderdenotes the definingclass loader of the current class.

For example, thefollowing code fragment returns the runtimeClassdescriptor for theclass namedjava.lang.Thread:

Class t =Class.forName("java.lang.Thread")

A call to forName("X")causes theclass named Xto beinitialized.

Parameters:

className- the fully qualifiedname of the desired class.

Returns:

theClassobject for the classwith the specified name.

從官方給出的API文件中可以看出:

Class.forName(className)實際上是呼叫Class.forName(className,true, this.getClass().getClassLoader())。第二個引數,是指Class被loading後是不是必須被初始化。可以看出,使用Class.forName(className)載入類時則已初始化。

所以Class.forName(className)可以簡單的理解為:獲得字串引數中指定的類,並初始化該類

一.首先你要明白在java裡面任何class都要裝載在虛擬機器上才能執行。

1.      forName這句話就是裝載類用的(new是根據載入到記憶體中的類建立一個例項,要分清楚)。

2.      至於什麼時候用,可以考慮一下這個問題,給你一個字串變數,它代表一個類的包名和類名,你怎麼例項化它?

           A a = (A)Class.forName("pacage.A").newInstance();這和 A a =new A();是一樣的效果。

3.      jvm在裝載類時會執行類的靜態程式碼段,要記住靜態程式碼是和class繫結的,class裝載成功就表示執行了你的靜態程式碼了,而且以後不會再執行這段靜態程式碼了。

4.      Class.forName(xxx.xx.xx)的作用是要求JVM查詢並載入指定的類,也就是說JVM會執行該類的靜態程式碼段。

5.      動態載入和建立Class 物件,比如想根據使用者輸入的字串來建立物件

       String str = 使用者輸入的字串  

       Class t = Class.forName(str);  

       t.newInstance(); 

二.在初始化一個類,生成一個例項的時候,newInstance()方法和new關鍵字除了一個是方法,一個是關鍵字外,最主要有什麼區別?

      1.它們的區別在於建立物件的方式不一樣,前者是使用類載入機制,後者是建立一個新類。

      2.那麼為什麼會有兩種建立物件方式?

        這主要考慮到軟體的可伸縮、可擴充套件和可重用等軟體設計思想。 
        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就可以。 
        3.從JVM的角度看,我們使用關鍵字new建立一個類的時候,這個類可以沒有被載入。  但是使用newInstance()方法的時候,

         就必須保證:

              1、這個類已經載入;

              2、這個類已經連線了。

        而完成上面兩個步驟的正是Class的靜態方法forName()所完成的,這個靜態方法呼叫了啟動類載入器,即載入 java API的那個載入器。 
         現在可以看出,newInstance()實際上是把new這個方式分解為兩步,即首先呼叫Class載入方法載入某個類,然後例項化。這樣分步的好處是顯而易見的。我們可以在呼叫class的靜態載入方法forName時獲得更好

         的靈活性,提供給了一種降耦的手段。 

三.最後用最簡單的描述來區分new關鍵字和newInstance()方法的區別: 

1. newInstance: 弱型別。低效率。只能呼叫無參構造。 
         2. new: 強型別。相對高效。能呼叫任何public構造。

3、應用情景:

情景一:載入資料庫驅動的時候

Class.forName的一個很常見的用法是在載入資料庫驅動的時候。

如:

[java] view plain copy  print?
  1. Class.forName("com.microsoft.sqlserver.jdbc.SQLServerDriver");  
  2. Connection con=DriverManager.getConnection("jdbc:sqlserver://localhost:1433;DatabaseName==JSP","jph","jph");      

  為什麼在我們載入資料庫驅動包的時候有的卻沒有呼叫newInstance( )方法呢?

即有的jdbc連線資料庫的寫法裡是Class.forName(xxx.xx.xx);而有一些:Class.forName(xxx.xx.xx).newInstance(),為什麼會有這兩種寫法呢?

剛才提到,Class.forName("");的作用是要求JVM查詢並載入指定的類,如果在類中有靜態初始化器的話,JVM必然會執行該類的靜態程式碼段。

而在JDBC規範中明確要求這個Driver類必須向DriverManager註冊自己,即任何一個JDBCDriver的Driver類的程式碼都必須類似如下:
          public classMyJDBCDriver implements Driver {

    static{

       DriverManager.registerDriver(new MyJDBCDriver());

   }

   }

  既然在靜態初始化器的中已經進行了註冊,所以我們在使用JDBC時只需要Class.forName(XXX.XXX);就可以了。

情景二:使用AIDL與電話管理Servic進行通訊

Method method =Class.forName("android.os.ServiceManager")

         .getMethod("getService",String.class);

// 獲取遠端TELEPHONY_SERVICEIBinder物件的代理

IBinder binder =(IBinder) method.invoke(null, new Object[] { TELEPHONY_SERVICE});

// IBinder物件的代理轉換為ITelephony物件

ITelephonytelephony = ITelephony.Stub.asInterface(binder);

// 結束通話電話

telephony.endCall();