天天看點

Class.forName()的作用與使用總結1、Class類簡介: 2、 Class.forName()方法: 3、應用情景:

      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)...

Class.forName:傳回與給定的字元串名稱相關聯類或接口的Class對象。

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

<code>static</code> <code>Class</code><code>&lt;?&gt;</code>

forName<code>(</code><code>String</code><code> </code><code>className)</code>

Returns the <code>Class</code> object associated with the class or interface with the given string name.

forName<code>(</code><code>String</code><code> </code><code>name, boolean initialize,</code> <code>ClassLoader</code><code> </code><code>loader)</code>

Returns the <code>Class</code> object associated with the class or interface with the given string name, using the given class loader.

說明:

publicstatic Class&lt;?&gt; forName(String className)

Returns the <code>Class</code> object associated withthe class or interface with the given string name.Invokingthis method is equivalent to:

<code>Class.forName(className,true, currentLoader)</code>

where <code>currentLoader</code> denotes the definingclass loader of the current class.

For example, thefollowing code fragment returns the runtime <code>Class</code> descriptor for theclass named <code>java.lang.Thread</code>:

<code>Class t =Class.forName("java.lang.Thread")</code>

A call to <code>forName("X")</code> causes theclass named <code>X</code> to beinitialized.

Parameters:

<code>className</code> - the fully qualifiedname of the desired class.

Returns:

the <code>Class</code> object 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構造。

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

如:

Class.forName("com.microsoft.sqlserver.jdbc.SQLServerDriver");  

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進行通信

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

// 擷取遠端TELEPHONY_SERVICE的IBinder對象的代理

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

// 将IBinder對象的代理轉換為ITelephony對象

ITelephonytelephony = ITelephony.Stub.asInterface(binder);

// 挂斷電話

telephony.endCall();

參考資料:

JDK1.8_API.../docs/api/java/lang/Class.html

http://www.ibm.com/developerworks/cn/java/j-lo-classloader/

    本文轉自 一點點征服   部落格園部落格,原文連結:http://www.cnblogs.com/ldq2016/p/6834865.html,如需轉載請自行聯系原作者