1. 程式人生 > >Java反射機制的原理及作用

Java反射機制的原理及作用

反射機制是Java特性之一,反射機制是構建框架技術的基礎所在。靈活掌握Java反射機制,對大家以後學習框架技術有很大的幫助。

那麼什麼是Java的反射呢?

       大家都知道,要讓Java程式能夠執行,那麼就得讓Java類要被Java虛擬機器載入。Java類如果不被Java虛擬機器載入,是不能正常執行的。現在我們執行的所有的程式都是在編譯期的時候就已經知道了你所需要的那個類的已經被載入了。

Java的反射機制是在編譯並不確定是哪個類被載入了,而是在程式執行的時候才載入、探知、自審。使用在編譯期並不知道的類。這樣的特點就是反射。

那麼Java反射有什麼作用呢?

假如我們有兩個程式設計師,一個程式設計師在寫程式的時候,需要使用第二個程式設計師所寫的類,但第二個程式設計師並沒完成他所寫的類。那麼第一個程式設計師的程式碼能否通過編譯呢?這是不能通過編譯的。利用Java反射的機制,就可以讓第一個程式設計師在沒有得到第二個程式設計師所寫的類的時候,來完成自身程式碼的編譯。

Java的反射機制它知道類的基本結構,這種對Java類結構探知的能力,我們稱為Java類的“自審”。大家都用過Jcreator和eclipse。當我們構建出一個物件的時候,去呼叫該物件的方法和屬性的時候。一按點,編譯工具就會自動的把該物件能夠使用的所有的方法和屬性全部都列出來,供使用者進行選擇。這就是利用了Java反射的原理,是對我們建立物件的探知、自審。

Class類

要正確使用Java反射機制就得使用java.lang.Class這個類。它是Java反射機制的起源。當一個類被載入以後,Java虛擬機器就會自動產生一個Class物件。通過這個Class物件我們就能獲得載入到虛擬機器當中這個Class物件對應的方法、成員以及構造方法的宣告和定義等資訊。

反射API

u反射API用於反應在當前Java虛擬機器中的類、介面或者物件資訊

u功能
—獲取一個物件的類資訊.

       —獲取一個類的訪問修飾符、成員、方法、構造方法以及超類的資訊.

       —檢獲屬於一個介面的常量和方法宣告.

       —建立一個直到程式執行期間才知道名字的類的例項.

       —獲取並設定一個物件的成員,甚至這個成員的名字是
   在程式執行期間才知道.

       —檢測一個在執行期間才知道名字的物件的方法

       利用Java反射機制我們可以很靈活的對已經載入到Java虛擬機器當中的類資訊進行檢測。當然這種檢測在對執行的效能上會有些減弱,所以什麼時候使用反射,就要靠業務的需求、大小,以及經驗的積累來決定。

       那麼如何利用反射API在執行的時候知道一個類的資訊呢?

程式碼示例:

[java] view plain copy print?
  1. <span style="font-size:16px;">import java.lang.reflect.Field;  
  2. import java.lang.reflect.Method;  
  3. import javax.swing.JOptionPane;  
  4. /** 
  5.   *本類用於測試反射API,利用使用者輸入類的全路徑, 
  6. *找到該類所有的成員方法和成員屬性 
  7.   */
  8. publicclass MyTest {  
  9.      /** 
  10.      *構造方法 
  11.      */
  12.     public MyTest(){  
  13.        String classInfo=JOptionPane.showInputDialog(null,"輸入類全路徑");//要求使用者輸入類的全路徑
  14.        try {  
  15.            Class cla=Class.forName(classInfo);//根據類的全路徑進行類載入,返回該類的Class物件
  16.            Method[] method=cla.getDeclaredMethods();//利用得到的Class物件的自審,返回方法物件集合
  17.            for(Method me:method){//遍歷該類方法的集合
  18.               System.out.println(me.toString());//列印方法資訊
  19.            }  
  20.            System.out.println("********");  
  21.            Field[] field=cla.getDeclaredFields();//利用得到的Class物件的自審,返回屬性物件集合
  22.            for(Field me:field){ //遍歷該類屬性的集合
  23.               System.out.println(me.toString());//列印屬性資訊
  24.            }  
  25.        } catch (ClassNotFoundException e) {  
  26.            e.printStackTrace();  
  27.        }  
  28.     }  
  29.     publicstaticvoid main(String[] args) {  
  30.        new MyTest();  
  31.     }  
  32. }</span>  
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import javax.swing.JOptionPane;
/**
  *本類用於測試反射API,利用使用者輸入類的全路徑,
*找到該類所有的成員方法和成員屬性
  */
public class MyTest {
     /**
     *構造方法
     */
    public MyTest(){
       String classInfo=JOptionPane.showInputDialog(null,"輸入類全路徑");//要求使用者輸入類的全路徑
       try {
           Class cla=Class.forName(classInfo);//根據類的全路徑進行類載入,返回該類的Class物件
          
           Method[] method=cla.getDeclaredMethods();//利用得到的Class物件的自審,返回方法物件集合
          
           for(Method me:method){//遍歷該類方法的集合
              System.out.println(me.toString());//列印方法資訊
           }
          
           System.out.println("********");
          
           Field[] field=cla.getDeclaredFields();//利用得到的Class物件的自審,返回屬性物件集合
           for(Field me:field){ //遍歷該類屬性的集合
              System.out.println(me.toString());//列印屬性資訊
           }
       } catch (ClassNotFoundException e) {
           e.printStackTrace();
       }
    }
    public static void main(String[] args) {
       new MyTest();
    }
}

執行的時候,我們輸入javax.swing.JFrame,那麼執行結果如下:

public void javax.swing.JFrame.remove(java.awt.Component)

public void javax.swing.JFrame.update(java.awt.Graphics)

…………

********

public static final int javax.swing.JFrame.EXIT_ON_CLOSE

private int javax.swing.JFrame.defaultCloseOperation

…………

    大家可以發現,類的全路徑是在程式執行的時候,由使用者輸入的。所以虛擬機器事先並不知道所要載入類的資訊,這就是利用反射機制來對使用者輸入的類全路徑來對類自身的一個自審。從而探知該類所擁有的方法和屬性。

通過上面程式碼,大家可以知道編譯工具為什麼能夠一按點就能列出使用者當前物件的屬性和方法了。它是先獲得使用者輸入物件的字串,然後利用反射原理來對這樣的類進行自審,從而列出該類的方法和屬性。

使用反射機制的步驟:

u匯入java.lang.relfect 包

u遵循三個步驟
第一步是獲得你想操作的類的 java.lang.Class 物件
第二步是呼叫諸如 getDeclaredMethods 的方法
第三步使用 反射API 來操作這些資訊

獲得Class物件的方法

u如果一個類的例項已經得到,你可以使用

       【Class c = 物件名.getClass(); 

      例: TextField t = new TextField();

              Class c = t.getClass();

              Class s = c.getSuperclass();

u如果你在編譯期知道類的名字,你可以使用如下的方法

Class c = java.awt.Button.class; 
或者

         Class c = Integer.TYPE;

u如果類名在編譯期不知道, 但是在執行期可以獲得, 你可以使用下面的方法

          Class c = Class.forName(strg);

   這樣獲得Class類物件的方法,其實是利用反射API把指定字串的類載入到記憶體中,所以也叫類載入器載入方法。這樣的話,它會把該類的靜態方法和靜態屬性,以及靜態程式碼全部載入到記憶體中。但這時候,物件還沒有產生。所以為什麼靜態方法不能訪問非靜態屬性和方法。因為靜態方法和屬性產生的時機在非靜態屬性和方法之前。

程式碼示例:

[java] view plain copy print?
  1. <span style="font-size:16px;">package  com;  
  2. publicclass MyTest {  
  3.     publicstaticvoid main(String[] args) {  
  4.        TestOne  one=null;  
  5.        try{  
  6.        Class  cla=Class.forName("com.TestOne");//進行com.TestOne類載入,返回一個Class物件
  7.        System.out.println("********");  
  8.        one=(TestOne)cla.newInstance();//產生這個Class類物件的一個例項,呼叫該類無參的構造方法,作用等同於new TestOne()
  9.        }catch(Exception e){  
  10.            e.printStackTrace();  
  11.        }  
  12.        TestOne two=new TestOne();  
  13.   System.out.println(one.getClass() == two.getClass());//比較兩個TestOne物件的Class物件是否是同一個物件,在這裡結果是true。說明如果兩個物件的型別相同,那麼它們會有相同的Class物件
  14.     }  
  15. }  
  16. class TestOne{  
  17.     static{  
  18.        System.out.println("靜態程式碼塊執行");  
  19.     }  
  20.     TestOne(){  
  21.        System.out.println("構造方法");  
  22.     }  
  23. }</span>  
package  com;
 
public class MyTest {
    public static void main(String[] args) {
       TestOne  one=null;
       try{
       Class  cla=Class.forName("com.TestOne");//進行com.TestOne類載入,返回一個Class物件
       System.out.println("********");
       one=(TestOne)cla.newInstance();//產生這個Class類物件的一個例項,呼叫該類無參的構造方法,作用等同於new TestOne()
       }catch(Exception e){
           e.printStackTrace();
       }
       TestOne two=new TestOne();
  System.out.println(one.getClass() == two.getClass());//比較兩個TestOne物件的Class物件是否是同一個物件,在這裡結果是true。說明如果兩個物件的型別相同,那麼它們會有相同的Class物件
    }
}
 
class TestOne{
    static{
       System.out.println("靜態程式碼塊執行");
    }
    TestOne(){
       System.out.println("構造方法");
    }
}

  以上程式碼過行的結果是:

靜態程式碼塊執行

***********

構造方法

構造方法

程式碼分析:

在進行Class.forName("com.TestOne")的時候,實際上是對com.TestOne進行類載入,這時候,會把靜態屬性、方法以及靜態程式碼塊都載入到記憶體中。所以這時候會打印出"靜態程式碼塊執行"。但這時候,物件卻還沒有產生。所以"構造方法"這幾個字不會列印。當執行cla.newInstance()的時候,就是利用反射機制將Class物件生成一個該類的一個例項。這時候物件就產生了。所以列印"構造方法"。當執行到TestOne two=new TestOne()語句時,又生成了一個物件。但這時候類已經載入完畢,靜態的東西已經載入到記憶體中,而靜態程式碼塊只執行一次,所以不用再去載入類,所以只會列印"構造方法",而"靜態程式碼塊執行"不會列印。

反射機制不但可以例出該類物件所擁有的方法和屬性,還可以獲得該類的構造方法及通過構造方法獲得例項。也可以動態的呼叫這個例項的成員方法。

程式碼示例:

[java] view plain copy print?
  1. <span style="font-size:16px;">package reflect;  
  2. import java.lang.reflect.Constructor;  
  3. /** 
  4.  * 
  5.  * 本類測試反射獲得類的構造器物件, 
  6.  * 並通過類構造器物件生成該類的例項 
  7.  * 
  8.  */
  9. publicclass ConstructorTest {  
  10.     publicstaticvoid main(String[] args) {  
  11.        try {  
  12.            //獲得指定字串類物件
  13.            Class cla=Class.forName("reflect.Tests");  
  14.            //設定Class物件陣列,用於指定構造方法型別
  15.            Class[] cl=new Class[]{int.class,int.class};  
  16.            //獲得Constructor構造器物件。並指定構造方法型別
  17.            Constructor con=cla.getConstructor(cl);  
  18.            //給傳入引數賦初值
  19.            Object[] x={new Integer(33),new Integer(67)};  
  20.            //得到例項
  21.            Object obj=con.newInstance(x);  
  22.        } catch (Exception e) {  
  23.            e.printStackTrace();  
  24.        }  
  25.     }  
  26. }  
  27. class Tests{  
  28.     public Tests(int x,int y){  
  29.        System.out.println(x+"    "+y);  
  30.     }  
  31. }</span>  
package reflect;
 
import java.lang.reflect.Constructor;
 
 
/**
 *
 * 本類測試反射獲得類的構造器物件,
 * 並通過類構造器物件生成該類的例項
 *
 */
public class ConstructorTest {
 
    public static void main(String[] args) {
       try {
           //獲得指定字串類物件
           Class cla=Class.forName("reflect.Tests");
           //設定Class物件陣列,用於指定構造方法型別
           Class[] cl=new Class[]{int.class,int.class};
          
           //獲得Constructor構造器物件。並指定構造方法型別
           Constructor con=cla.getConstructor(cl);
          
           //給傳入引數賦初值
           Object[] x={new Integer(33),new Integer(67)};
          
           //得到例項
           Object obj=con.newInstance(x);
       } catch (Exception e) {
           e.printStackTrace();
       }
    }
 
}
 
class Tests{
    public Tests(int x,int y){
       System.out.println(x+"    "+y);
    }
}

執行的結果是” 33    67。說明我們已經生成了Tests這個類的一個物件。

相關推薦

java反射原理使用

int port mar unit .class 使用方法 www. fec pos 定義 JAVA反射機制是在運行狀態中,對於任意一個類,都能夠知道這個類的所有屬性和方法;對於任意一個對象,都能夠調用它的任意方法和屬性;這種動態獲取信息以及動態調用對象方法的

Java 反射機制原理

反射的概念 反射的概念是由Smith在1982年首次提出的,主要是指程式可以訪問、檢測和修改它本身狀態或行為的一種能力。這一概念的提 出很快引發了電腦科學領域關於應用反射性的研究。它首先被程式語言的設計領域所採用,並在Lisp和麵向物件方面取得了成績。其中 LEAD/LEAD++ 、Open

java反射原理作用

什麼是反射,反射原理 Java反射的原理:java類的執行需要經歷以下過程, 編譯:.java檔案編譯後生成.class位元組碼檔案 載入:類載入器負責根據一個類的全限定名來讀取此類的二進位制位元組流到JVM內部,並存儲在執行時記憶體區的方法區,然後將其轉換為一個與目標

Java反射機制原理作用

反射機制是Java特性之一,反射機制是構建框架技術的基礎所在。靈活掌握Java反射機制,對大家以後學習框架技術有很大的幫助。 那麼什麼是Java的反射呢?        大家都知道,要讓Java程式能夠執行,那麼就得讓

Java反射機制原理

一、概念 java程式執行時動態的建立類並呼叫類的方法和屬性   二、原理簡介 Class<?> clz = Class.forName("java.util.ArrayList"); ArrayList object = (ArrayLis

Java反射機制原理在Android下的簡單應用

package crazypebble.reflectiontest;import java.lang.reflect.Constructor;import java.lang.reflect.Method;publicclass LoadMethod { /** * 在執行時載入指定的類,並呼

【原】Java反射機制原理在Android下的簡單應用

轉載地址:http://www.cnblogs.com/crazypebble/archive/2011/04/13/2014582.html   花了幾天時間,研究了一下Java的反射機制。在這裡總結一下這幾天學習的成果,一來分享自己的學習過程

JAVA反射機制CLASS.FORNAME的作用含義

最近由於工作上需要,對reflection做了一番瞭解,以下是學習總結,有不少內容是借鑑的,但已無法找到源文出處,還請原文作者見諒。 Reflection 是Java被視為動態(或準動態)語言的一個關鍵性質。這個機制允許程式在執行時透過Reflection APIs取得任

java反射機制原理

動態調用 clas java語言 動態 方法 gpo mod odi log 一 反射機制的概念: 指在運行狀態中,對於任意一個類,都能夠知道這個類的所有屬性和方法,對於任意一個對象,都能調用它的任意一個方法.這種動態獲取信息,以及動態調用對象方法的功能叫java語言的反

Java反射機制作用

機制 解答 java 構造 調用 ava java反射機制 成員 方法 Java反射機制的作用? 解答:Java反射機制的作用是: 1)在運行時判斷任意一個對象所屬的類。 2)在運行時構造任意一個類的對象。 3)在運行時判斷任意一個類所具有的成員變量和方法。

Java反射機制getModifiers()方法的作用

JAVA 反射機制中,Field的getModifiers()方法返回int型別值表示該欄位的修飾符。 其中,該修飾符是java.lang.reflect.Modifier的靜態屬性。 對應表如下: PUBLIC: 1 PRIVATE: 2 PROTECTED: 4 STATI

02、java反射機制的核心原理

一、java的核心機制 java有兩種核心機制:java虛擬機器(JavaVirtual Machine)與垃圾收集機制(Garbage collection): 1、Java虛擬機器:是執行所有Java程式的抽象計算機,是Java語言的執行環境,在其上面執行Java程式碼編譯後的位元組碼程

Java中的類載入和Class.forName();java反射機制原理

對於大部分人來說,第一次見到class.forName(String className)這句程式碼應該是在使用jdbc方式連線資料庫的時候。但這句程式碼本質上是什麼含義,做了什麼工作呢?本文將回答此問題。 理解Class.forName方法需要一些知識鋪墊,也就是

Java集合-05fail-fast(快速失敗)機制原理解決方法

fail-fast簡介 fail-fast(快速失敗),是Java集合的一種錯誤檢測機制。當在遍歷集合的過程中該集合在結構(改變集合大小)上發生變化時候, 有可能發生fail-fast,丟擲java.util.ConcurrentModificationException異常。 fail-fas

hibernate工作原理作用 JAVA Hibernate工作原理為什麼要用

轉載自 http://www.cnblogs.com/dashi/p/3597969.html#commentform JAVA Hibernate工作原理及為什麼要用 hibernate 簡介:hibernate是一個開源框架,它是物件關聯關係對映的框架,它對JDBC做了輕量級的封裝,而我們j

這是轉載的 java反射機制原理,原作者講解的相對很通俗易懂

轉自這裡 Java的反射機制是Java特性之一,反射機制是構建框架技術的基礎所在。靈活掌握Java反射機制,對大家以後學習框架技術有很大的幫助。 那麼什麼是Java的反射呢?        大

JAVA反射機制作用是什麼

一、什麼是反射: 反射的概念是由Smith在1982年首次提出的,主要是指程式可以訪問、檢測和修改它本身狀態或行為的一種能力。這一概念的提出很快引發了電腦科學領域關於應用反射性的研究。它首先被程式語言的設計領域所採用,並在Lisp和麵向物件方面取得了成績。其中LEAD/LEA

Java中this關鍵字原理作用

this關鍵字 一:this關鍵字的作用 *當方法的區域性變數和類的變數名重名時,在不使用this的情況下會有什麼結果? Person String name;//定義當前Person名字 //引數name是目標名字 //成員變數name是自己名字 public void s

Java 反射機制和動態代理是基於什麼原理,瞭解過嗎?

工作多年以及在面試中,我經常能體會到,有些面試者確實是認真努力工作,但坦白說表現出的能力水平卻不足以通過面試,通常是兩方面原因: 1、“知其然不知其所以然”。 做了多年技術,開發了很多業務應用,但似乎並未思考過種種技術選擇背後的邏輯。坦白說,我並不放心把具有一定深度的任務交給他。 2、

Java反射機制詳解 Method.invoke解釋

JAVA反射機制 JAVA反射機制是在執行狀態中,對於任意一個類,都能夠知道這個類的所有屬性和方法;對於任意一個物件,都能夠呼叫它的任意一個方法;這種動態獲取的資訊以及動態呼叫物件的方法的功能稱為java語言的反射機制。 Java反射機制主要提供了以下功能: 在執行時判