首頁 >Java >java教程 >Java反射機制原理實例分析

Java反射機制原理實例分析

WBOY
WBOY轉載
2023-05-19 21:13:041057瀏覽

什麼是反射?

反射機制是在運作狀態中,它為Java提供一種「操作對象」的能力,在運作狀態下,透過Class檔案對象,可以呼叫到任何類別裡面的屬性、方法、以及建構方法,包括私有的,所有的類別在反射機制面前都是透明的

自己的概括:透過Class檔案物件可以看到這個類別裡面的所有東西,並且可以使用和修改

Java反射機制原理實例分析

#反射的前提是取得Class檔案物件((字節碼物件),那麼一共有三種方式取得:

  • Class.forName (「全類別名稱」) ----透過Class類別的靜態方法(最常用)

  • 類別名稱.class

  • 對象.getClass()

//方式1:获取字节码对象,Class.forName("全类名")
Class cla1 = Class.forName("Study01.Person");

//方式2: 类名.Class
Class cla2 = Person.class;

//方式3:对象.getClass();
Person per = new Person();
Class cla3 = per.getClass();

//这三个class对象都是由Person这个类生成的
//那么我们看一下这三个字节码对象是不是同一个:

System.out.println(cla1==cla2);
System.out.println(cla2==cla3);

//輸出結果:  兩個true

結論:

  • 字節碼物件在類別載入的時候就產生,並且只有一個

  • 無論哪種方式取得字節碼物件都是同一個字節碼物件

#透過反射來取得類別中的屬性:

取得到Class字節碼物件後,我們就可以透過字節碼物件來取得我們想要取得的類別的屬性、方法、建構方法、以及private修飾的。

部分Class方法:

Java反射機制原理實例分析

Demo示範:

#1、建構一個Person類,裡面有兩個public和兩個private的屬性(不設定建構和get/set,就是看反射能不能得到裡面的值)

public class Person {

    private String name;  //名字
    private int age = 18;   //年龄

    public int ID = 123;   //身份证
    public String Sex;  //性别

    @Override
    public String toString(){
        return "姓名"+name+"年龄:"+age+"ID:"+ID+"性别:"+Sex;
    }
}

測試類別:

public class Test {

    public static void main(String[] args) throws ClassNotFoundException {

		//获取Class文件对象,用最常用的通过Class类的静态方法
        Class per = Class.forName("Test01.Person"); //这里是传入全路径!!从最外层的包名开始!
        //使用getFields()方法获取全部被public修饰的属性(方法上面的截图有)
        //并且返回的是Field类型的数组
        Field fields[] = per.getFields();
        for (Field field:fields) {
            System.out.println(field);
        }
    }
}

輸出:

我們成功的取得到了Person類別中全部public屬性

Java反射機制原理實例分析

2、也可以取得全部的屬性,包括私有的:(其他程式碼就不重寫啦)

 for (Field field : per.getDeclaredFields()) {
            System.out.println(field);
        }

輸出:

Java反射機制原理實例分析

#3、取得公有的屬性,並且修改這個值:

	
	Field f = per.getField("Sex");
	System.out.println(f);
	
	//获取一个对象:
	Object obj = per.getConstructor().newInstance();
	//修改值:
	f.set(obj,"男");
	Person p = (Person)obj;
	
	System.out.println(p.Sex);

輸出:

Java反射機制原理實例分析

4、取得私有的屬性,並且修改這個值: 這裡把上面修改公有屬性的值也連起來:

   
        Person p = (Person)obj;
        
        //获取公有字段并调用,并修改
        Field f = per.getField("Sex");
        //获取一个对象:
        Object obj = per.getConstructor().newInstance();
        f.set(obj,"男");  //将Sex的属性修改成了 男

        //调用私有的属性,并修改
        f = per.getDeclaredField("name");
        //在访问私有的属性的值之前,先要设置运行访问↓
        //在访问之前忽略访问权限的检查,叫暴力反射
        f.setAccessible(true);
        f.set(obj,"张三");
        System.out.println("Person里面的信息是:"+p.toString());
    }
}

輸出:

Java反射機制原理實例分析

透過反射來取得類別中的方法(公有、私有、建構):

Person類別:

public class Person {

    private String name;  //名字
    private int age = 18;   //年龄
    public int ID = 123;   //身份证
    public String Sex ;  //性别
    
    //构造:
    public Person() {}

    public Person(String name, int age, int ID, String sex) {
        this.name = name;
        this.age = age;
        this.ID = ID;
        Sex = sex;
    }
    
    //无参公有方法:
    public void eat(){
        System.out.println("我会吃饭");
    }
    
    //有参公有方法:
    public void eat(String food){
        System.out.println("我在吃:"+food);
    }
    
    //有参私有方法
    private void  play(String name){
        System.out.println(name+"在玩");
    }
}

測試類別:

public class Test {

    public static void main(String[] args) throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {

        //获取到Person以及父类Object里面的public方法:
        System.out.println("-----获取到Person以及父类Object里面的public方法↓-----");
        for (Method method : Person.class.getMethods()) {
            System.out.println(method);
            System.out.println("方法名:"+ method.getName());
        }

        //获取到Person里面的方法,包括私有
        System.out.println("-----获取到Person里面的方法,包括私有↓-----");
        for (Method method:Person.class.getDeclaredMethods()) {
            System.out.println(method.getName()+"  ");
        }

        //按照方法名获取到Person中的eat方法:
        System.out.println("-----根据方法名获取到Person类中的eat方法↓-----");
        Method earMethod1 = Person.class.getMethod("eat");
        Person per = new Person();
        //通过invoke(Object,param...)来调用指定的方法
        earMethod1.invoke(per);

        //使用反射调用有参方法;
        System.out.println("-----使用反射调用有参方法(传入参数)↓-----");
        Method earMethod2 = Person.class.getMethod("eat",String.class);
        earMethod2.invoke(per,"牛肉");

        //通过暴力反射获取到私有的play方法:
        System.out.println("-----通过暴力反射获取到私有的play方法传入参数)↓-----");
        Method earMethod3 = Person.class.getDeclaredMethod("play", String.class);
        //在访问私有的属性的方法之前,先要设置运行访问
        earMethod3.setAccessible(true);
        earMethod3.invoke(per,"小王");

    }

輸出:

-----取得到Person以及父類別Object裡面的public方法↓-----
public void Test02.Person.eat(java .lang.String)
方法名稱:eat
public void Test02.Person.eat()
方法名稱:eat
public final void java.lang.Object.wait() throws java.lang .InterruptedException
方法名稱:wait
public final void java.lang.Object.wait(long,int) throws java.lang.InterruptedException
方法名稱:wait
public final native void java.lang .Object.wait(long) throws java.lang.InterruptedException
方法名稱:wait
public boolean java.lang.Object.equals(java.lang.Object)
方法名稱:equals
public java.lang.String java.lang.Object.toString()
方法名稱:toString
public native int java.lang.Object.hashCode()
方法名稱:hashCode
public final native java .lang.Class java.lang.Object.getClass()
方法名稱:getClass
public final native void java.lang.Object.notify()
方法名稱:notify
public final native void java.lang.Object.notifyAll()
方法名稱:notifyAll

以上是Java反射機制原理實例分析的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文轉載於:yisu.com。如有侵權,請聯絡admin@php.cn刪除