關於Java反射的理解


title: 關於Java反射的理解
date: 2017-12-07 14:34:21
tags: javaSE
categories: javaSE

首先,從動態語言講起。像Python、Ruby這種語言,只要修改了代碼,修改的效果立即生效,因爲這種語言是無需編譯,直接執行代碼的,我們稱這類語言是“動態語言”。而C++、Java這種,在運行之前需要先編譯,如果中途修改了代碼不重新編譯去執行的話就沒有變化。但是,Java有一個非常突出的動態相關機制,即反射:我們可以於運行時(區別於編譯時)加載、探知、使用編譯期間完全未知的classes。換句話說,Java程序可以加載一個運行時才得知名稱的class(在這之前修改這個類即時不編譯都有效),獲悉其完整構造(但不包括methods定義),並生成其對象實體、或對其fields設值、或喚起其methods。

再通俗地說一下什麼是反射?

普通的Java對象是通過new關鍵字把對應類的字節碼文件加載到內存,然後創建該對象的。

反射是通過一個名爲Class的特殊類,用Class.forName(“類名”);得到類的字節碼對象,然後用newInstance()方法在虛擬機內部構造這個對象(針對無參構造函數)。

也就是說反射機制讓我們可以在程序運行時動態地拿到Java類對應的字節碼對象(而不是在編譯的時候),然後動態的進行任何可能的操作。反射的功能主要包括:

  • 在運行時判斷任意一個對象所屬的類
  • 在運行時構造任意一個類的對象
  • 在運行時判斷任意一個類所具有的成員變量和方法
  • 在運行時調用任意一個對象的方法(這樣就可以修改這個對象的屬性)
  • 生成動態代理

使用反射的主要作用是方便程序的擴展,由於其運行時動態加載的特性。

Class類

Java中只有2種東西不是面向對象的,一個基本類型,一個靜態成員(方法、變量、常量)。我們提供的每一個類也是對象,一個類的類類型是java.lang.Class類的實例對象

// 創建類Foo的實例對象
Foo foo1 = new Foo();

// Foo這個類也是一個實例對象,Class類的實例對象
// 任何一個類都是Class的實例對象,這個實例對象有三種表達方式

// 第一種表達方式。實際上表明任何一個類都有一個隱含的靜態成員變量Class
Class c1 = Foo.class;

// 第二種表達方式
Class c2 = foo1.getClass();

// 根據官網說法,c1,c2表示了Foo類的類類型(class type)
// 類也是對象,是Class類的實例對象,這個對象我們成爲該類的類類型

System.out.println(c1==c2)   // true
 
// 第三種表達方式
Class c3 = Class.forName("Foo");
 
System.out.println(c1==c3)   // true

Class類的構造器是私有的,只能JVM能創建Class類的實例對象。

可以通過類類型 (上面的c1 c2 c3)創建Foo類的實例對象:

Foo foo = (Foo)c1.newInstance();

動態加載類

什麼是動態加載?什麼是靜態加載?

靜態加載的類在編譯的時候就要提供,而動態加載的類在源程序編譯時可以缺席。區分編譯時和運行時。

Class.forName(“類名”) 這種方式,不僅表示了類的類類型,還代表了動態加載類。

用new這種方式靜態加載方式,編譯的時候,如果new的對象的那個類不存在的話,編譯不通過;但是用Class.forName這種動態加載方式,沒有這個類編譯的時候不會報任何錯,但是運行的時候會因爲找不到這個類而報錯。動態加載有什麼好處呢?配合接口編程,可以實現一個接口對多種實現,從而可以動態地去選擇完成不同的功能。因此,類動態加載對擴展功能很有用

其實,動態類加載主要就是通過反射機制將類對象注入進去。

靜態加載:

public class Office_Static {  
    public static void main(String[] args) {  
        //new 創建對象,是靜態加載類,在編譯時刻就需要用到Word和Excel,並將其編譯
        if("Word".equals(args[0])){  
            Word w = new Word();  
            w.start();  
        }  
        if("Excel".equals(args[0])){  
            Excel e = new Excel();  
            e.start();  
        }  
    }  
}

動態加載:

public interface OfficeAble {  
    public void start();  
}
public class Word implements OfficeAble {  
    public void start(){  
        System.out.println("word start");  
    }  
}
public class Excel implements OfficeAble {  
    public void start(){  
        System.out.println("excel start");  
    }  
}
public class OfficeBetter {  
    public static void main(String[] args) {  
        try {  
            //動態加載類,在運行時刻纔要用這個類  
            Class c = Class.forName(args[0]);//在運行配置裏面輸入com.imooc.加載類.Excel  
            //通過類類型,創建該類對象(先轉換爲Word和Excel的共同接口OfficeAble)  
            OfficeAble oa = (OfficeAble)c.newInstance();  
            oa.start();  
        } catch (Exception e) {  
            e.printStackTrace();  
        }  
    }  
  
}

通過反射動態獲取對象的方法屬性構造器信息

import java.lang.reflect.Method;
import java.lang.reflect.Field;
import java.lang.reflect.Constructor;

public class ClassUtil {
    /**
     * 打印類的信息,包括類的成員函數、成員變量(只獲取成員函數)
     * 
     * @param obj
     *            該對象所屬類的信息
     */
    public static void printClassMethodMessage(Object obj) {
        // 要獲取類的信息 首先要獲取類的類類型
        Class c = obj.getClass();// 傳遞的是哪個子類的對象 c就是該子類的類類型
        // 獲取類的名稱
        System.out.println("類的名稱是:" + c.getName());
        /*
         * Method類,方法對象 一個成員方法就是一個Method對象
         * getMethods()方法獲取的是所有的public的函數,包括父類繼承而來的
         * getDeclaredMethods()獲取的是所有該類自己聲明的方法,不問訪問權限
         */
        Method[] ms = c.getMethods();// c.getDeclaredMethods()
        for (int i = 0; i < ms.length; i++) {
            // 得到方法的返回值類型的類類型
            Class returnType = ms[i].getReturnType();
            System.out.print(returnType.getName() + " ");
            // 得到方法的名稱
            System.out.print(ms[i].getName() + "(");
            // 獲取參數類型--->得到的是參數列表的類型的類類型
            Class[] paramTypes = ms[i].getParameterTypes();
            for (Class class1 : paramTypes) {
                System.out.print(class1.getName() + ",");
            }
            System.out.println(")");
        }
    }

    /**
     * 獲取成員變量的信息
     * 
     * @param obj
     */
    public static void printFieldMessage(Object obj) {
        Class c = obj.getClass();
        /*
         * 成員變量也是對象 java.lang.reflect.Field Field類封裝了關於成員變量的操作
         * getFields()方法獲取的是所有的public的成員變量的信息
         * getDeclaredFields獲取的是該類自己聲明的成員變量的信息
         */
        // Field[] fs = c.getFields();
        Field[] fs = c.getDeclaredFields();
        for (Field field : fs) {
            // 得到成員變量的類型的類類型
            Class fieldType = field.getType();
            String typeName = fieldType.getName();
            // 得到成員變量的名稱
            String fieldName = field.getName();
            System.out.println(typeName + " " + fieldName);
        }
    }

    /**
     * 打印對象的構造函數的信息
     * 
     * @param obj
     */
    public static void printConMessage(Object obj) {
        Class c = obj.getClass();
        /*
         * 構造函數也是對象 java.lang. Constructor中封裝了構造函數的信息
         * getConstructors獲取所有的public的構造函數 getDeclaredConstructors得到所有的構造函數
         */
        // Constructor[] cs = c.getConstructors();
        Constructor[] cs = c.getDeclaredConstructors();
        for (Constructor constructor : cs) {
            System.out.print(constructor.getName() + "(");
            // 獲取構造函數的參數列表--->得到的是參數列表的類類型
            Class[] paramTypes = constructor.getParameterTypes();
            for (Class class1 : paramTypes) {
                System.out.print(class1.getName() + ",");
            }
            System.out.println(")");
        }
    }
}

通過反射運行時動態調用對象的方法

方法對象進行反射操作,運行時動態調用一個對象的方法:

import java.lang.reflect.Method;

public class MethodDemo1 {
    public static void main(String[] args) {
        // 要獲取print(int ,int )方法 1.要獲取一個方法就是獲取類的信息,獲取類的信息首先要獲取類的類類型
        A a1 = new A();
        Class c = a1.getClass();
        /*
         * 2.獲取方法 名稱和參數列表來決定 getMethod獲取的是public的方法 getDelcaredMethod自己聲明的方法
         */
        try {
            // Method m = c.getMethod("print", new
            // Class[]{int.class,int.class});
            Method m = c.getMethod("print", int.class, int.class);

            // 方法的反射操作
            // a1.print(10, 20);方法的反射操作是用m對象來進行方法調用 和a1.print調用的效果完全相同
            // 方法如果沒有返回值返回null,有返回值返回具體的返回值
            // Object o = m.invoke(a1,new Object[]{10,20});
            Object o = m.invoke(a1, 10, 20);
            System.out.println("==================");
            // 獲取方法print(String,String)
            Method m1 = c.getMethod("print", String.class, String.class);
            // 用方法進行反射操作
            // a1.print("hello", "WORLD");
            o = m1.invoke(a1, "hello", "WORLD");
            System.out.println("===================");
            // Method m2 = c.getMethod("print", new Class[]{});
            Method m2 = c.getMethod("print");
            // m2.invoke(a1, new Object[]{});
            m2.invoke(a1);
        } catch (Exception e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }

    }
}

class A {
    public void print() {
        System.out.println("helloworld");
    }

    public void print(int a, int b) {
        System.out.println(a + b);
    }

    public void print(String a, String b) {
        System.out.println(a.toUpperCase() + "," + b.toLowerCase());
    }
}

通過反射了解Java泛型的本質

反射的操作都是編譯之後的操作,反射動態加載的類是在程序運行時編譯並加載的。來看看下面這個例子。

import java.lang.reflect.Method;
import java.util.ArrayList;

public class MethodDemo2{
    public static void main(String[] args) {
        ArrayList list = new ArrayList();
        
        ArrayList<String> list1 = new ArrayList<String>();
        list1.add("hello");
        //list1.add(20);錯誤的
        Class c1 = list.getClass();
        Class c2 = list1.getClass();
        System.out.println(c1 == c2);// 都是ArrayList的類類型,true
        //反射的操作都是編譯之後的操作
        
        /*
         * c1==c2結果返回true說明編譯之後集合的泛型是去泛型化的
         * Java中集合的泛型,是防止錯誤輸入的,只在編譯階段有效,
         * 繞過編譯就無效了
         * 驗證:我們可以通過方法的反射來操作,繞過編譯
         */
        try {
            Method m = c2.getMethod("add", Object.class);
            m.invoke(list1, 20);//繞過編譯操作就繞過了泛型
            System.out.println(list1.size());
            System.out.println(list1);
            /*for (String string : list1) {
                System.out.println(string);
            }*///現在不能這樣遍歷
        } catch (Exception e) {
          e.printStackTrace();
        }
    }

}

轉載自:http://lioncruise.github.io/2016/11/29/java-reflection/

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章