1. 程式人生 > >關於Java的反射機制,你需要理解這些...

關於Java的反射機制,你需要理解這些...

反射機制是在執行狀態中,對於任意一個類,都能夠知道這個類的所有屬性和方法;對於任意一個物件,都能夠呼叫它的任意一個方法和屬性;這種動態獲取的資訊以及動態呼叫物件的方法的功能稱為java語言的反射機制。【翻譯於 官方文件】

本篇將從以下幾個方面講述反射的知識:

  • class 的使用
  • 方法的反射
  • 建構函式的反射
  • 成員變數的反射

一、什麼是class類

在面向物件的世界裡,萬物皆物件。類是物件,類是java.lang.Class類的例項物件。另外class類只有java虛擬機器才能new出來。任何一個類都是Class 類的例項物件。這例項物件有三種表達方式:

public class User{
}

public
class ClassTest{ User u=new User(); //方式1: Class c1=User.class; //方式2: Class c2=u.getClass(); //方式3: Class c3=Class.forName("com.forezp.User"); //可以通過類的型別建立該類的例項物件 User user=(User)c1.newInstance(); }

二、class類的動態載入

Class.forName(類的全稱);該方法不僅表示了類的型別,還代表了動態載入類。編譯時刻載入類是靜態載入、執行時刻載入類是動態載入類。

三、獲取方法資訊

基本的資料型別,void關鍵字都Class 類的例項;可以通過get
ame();getSimpleName()獲取類的名稱。

Class c1=String.class;
Class c2=int.class;
Class c3=void.class;
System.out.println(c1.getName());
System.out.println(c2.getSimpleName());

獲取類的所有方法,並打印出來:

public static void printClassInfo(Object object){
        Class c=object.getClass();
        System.out.println("類的名稱:"+c.getName());

        /**
         * 一個成員方法就是一個method物件
         * getMethod()所有的 public方法,包括父類繼承的 public
         * getDeclaredMethods()獲取該類所有的方法,包括private ,但不包括繼承的方法。
         */
Method[] methods=c.getMethods();//獲取方法 //獲取所以的方法,包括private ,c.getDeclaredMethods(); for(int i=0;i<methods.length;i++){ //得到方法的返回型別 Class returnType=methods[i].getReturnType(); System.out.print(returnType.getName()); //得到方法名: System.out.print(methods[i].getName()+"("); Class[] parameterTypes=methods[i].getParameterTypes(); for(Class class1:parameterTypes){ System.out.print(class1.getName()+","); } System.out.println(")"); } }
public classReflectTest{

        publicstaticvoidmain(String[] args){
                String s="ss";
                ClassUtil.printClassInfo(s);
        }
}

執行:

類的名稱:java.lang.String

booleanequals(java.lang.Object,)

java.lang.StringtoString()

inthashCode()

...

四、獲取成員變數的資訊

也可以獲取類的成員變數資訊


 public static void printFiledInfo(Object o){

        Class c=o.getClass();
        /**
         * getFileds()獲取public
         * getDeclaredFields()獲取所有
         */
        Field[] fileds=c.getDeclaredFields();

        for(Field f:fileds){
            //獲取成員變數的型別
            Class filedType=f.getType();
            System.out.println(filedType.getName()+" "+f.getName());
        }

    }

 public static void main(String[] args){
                String s="ss";
                //ClassUtil.printClassInfo(s);
                ClassUtil.printFiledInfo(s);
        }

執行:

[C value
int hash
long serialVersionUID
[Ljava.io.ObjectStreamField; serialPersistentFields
java.util.Comparator CASE_INSENSITIVE_ORDER
int HASHING_SEED
int hash32

五、獲取建構函式的資訊

public static void printConstructInfo(Object o){
        Class c=o.getClass();

        Constructor[] constructors=c.getDeclaredConstructors();
        for (Constructor con:constructors){
            System.out.print(con.getName()+"(");

            Class[] typeParas=con.getParameterTypes();
            for (Class class1:typeParas){
                System.out.print(class1.getName()+" ,");
            }
            System.out.println(")");
        }
    }

 public static void main(String[] args){
                String s="ss";
                //ClassUtil.printClassInfo(s);
                //ClassUtil.printFiledInfo(s);
                ClassUtil.printConstructInfo(s);
        }

執行:

java.lang.String([B ,)
java.lang.String([B ,int ,int ,)
java.lang.String([B ,java.nio.charset.Charset ,)
java.lang.String([B ,java.lang.String ,)
java.lang.String([B ,int ,int ,java.nio.charset.Charset ,)
java.lang.String(int ,int ,[C ,)
java.lang.String([C ,boolean ,)
java.lang.String(java.lang.StringBuilder ,)
java.lang.String(java.lang.StringBuffer ,)

...

六、方法反射的操作

獲取一個方法:需要獲取方法的名稱和方法的引數才能決定一個方法。

方法的反射操作:

method.invoke(物件,引數列表);

舉個例子:

class A{

    publicvoidadd(int a,int b){
        System.out.print(a+b);
    }

    publicvoidtoUpper(String a){
        System.out.print(a.toUpperCase());
    }
}

 public static void main(String[] args) {
        A a=new A();
        Class c=a.getClass();
        try {
            Method method=c.getMethod("add",new Class[]{int.class,int.class});
            //也可以 Method method=c.getMethod("add",int.class,int.class);
            //方法的反射操作
            method.invoke(a,10,10);
        }catch (Exception e){
            e.printStackTrace();
        }
    }

執行:

20

本篇文章已經講解了java反射的基本用法, 它可以在執行時判斷任意一個物件所屬的類;在執行時構造任意一個類的物件;在執行時判斷任意一個類所具有的成員變數和方法;在執行時呼叫任意一個物件的方法;生成動態代理。

我有一個微信公眾號,經常會分享一些Java技術相關的乾貨。如果你喜歡我的分享,可以用微信搜尋“Java團長”或者“javatuanzhang”關注。