首頁 > 軟體

Java反射機制詳解

2022-07-30 18:05:49

類的宣告週期

java原始碼----->javac-------------->java位元組碼檔案-------------->java----------------->類物件(所在記憶體空間:元空間,本地記憶體)------------------------new--------->範例化物件------------------gc------------->解除安裝物件

不同階段都可以獲取類物件

  • 物件.getClass() (記憶體階段)
  • Test.class (元空間)
  • class.forName(“類全名:包名+類名”) :(硬碟)都沒有進入記憶體空間就可以拿到物件

例如:運算元據庫時jdbc用到,還沒有進入記憶體之前,通過類全名,包名+類名,先把這個類給調出來使用,

獲取Class類物件的方式的場景

  • Class.forName(“類全名”) :多用於組態檔,將類名定義在組態檔中,讀取組態檔,載入類
  • 類名.class : 多用於引數的傳遞
  • 物件名.getClass():多用於物件獲取類物件

總結:同一個類載入器載入的檔案在一次程式執行過程中,只會被載入一次,無論使用哪種方式獲得的類物件都是同一個

程式碼範例:

package com.reflect;
public class TestReflectPerson {
    public static void main(String[] args) throws ClassNotFoundException {
        //1.class.forName()
        Class class1=Class.forName("com.reflect.Person");
        System.out.println(class1);
        //2.類名.class
        Class class2=Person.class;
        System.out.println(class2);
        //2.物件名.getClass()
        Class class3=new Person().getClass();
        System.out.println(class3);
        System.out.println(class1==class2);  //true
        System.out.println(class2==class3);  //true
    }
}

class類物件的功能

獲取成員變數 : 取所有:類物件.getDeclaredFields() ,獲取一個:類物件.getDeclaredField()

  • 設定值 set(Object obj,Object value)
  • 獲取值 get(Object obj)

獲取任意許可權修飾的成員變數獲取設定值,需要使用setAccessible(true)-----暴力反射

成員方法: 類物件.getDeclaredMethods()

執行方法 invoke(Object object ,Object… agrs) (引數個數任意,可有可無)

獲取方法名getName()

構造方法: 類物件.getDeclaredConstructors()

建立物件 newInstance() 優點:省掉獲取構造方法得到物件那一步,但是必須要有無參構造方法

該方法需要實際情況構造方法賦實參

//獲得構造方法物件,
        Constructor cons1 = pcla.getDeclaredConstructor(String.class, int.class);
        Person p2 = (Person)cons1.newInstance("李四",19);
        System.out.println("p2:"+p2.getName());

newInstance()如果是建立無參構造方法去建立物件,可以使用類物件來建立物件,跳過獲得構造方法物件

獲取

獲得類的名稱:getName() 列印出全名:類名+包名

只想列印單獨類名:getSimpleName()

獲取類的成員變數名稱

屬性檔案:內容以等號連線形如k=v,

程式碼範例:

package com.reflect;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
public class TestReflectPerson {
    public static void main(String[] args) throws Exception {
       /* //1.class.forName()
        Class class1=Class.forName("com.reflect.Person");
        System.out.println(class1);
        //2.類名.class
        Class class2=Person.class;
        System.out.println(class2);
        //2.類名.getClass()
        Class class3=new Person().getClass();
        System.out.println(class3);
        System.out.println(class1==class2);
        System.out.println(class2==class3);*/
        //獲取物件
        Class tclass=Class.forName("com.reflect.Person");
        //通過類物件獲取成員變數們
        Field[] fields = tclass.getDeclaredFields();
        System.out.println("獲取Person物件的所有屬性物件");
        for (Field field:fields){
           System.out.println(field);
       }
        //指定獲取Person物件的屬性物件
        System.out.println("指定獲取Person物件的屬性物件");
        Field age=tclass.getDeclaredField("age");
        System.out.println("age:"+age);
        //通過類物件獲取所有的構造方法
        Constructor[] constructors = tclass.getDeclaredConstructors();
        System.out.println("獲取Person的所有構造方法物件");
        for (Constructor constructor:constructors){
            System.out.println(constructor);
        }
        //通過類物件獲取無參的構造方法
        Constructor constructor = tclass.getDeclaredConstructor();
        System.out.println("constructor:"+constructor);
        //通過類物件獲取有參的構造方法
        Constructor constructor1 = tclass.getDeclaredConstructor(String.class,int.class);
        System.out.println("constructor1:"+constructor1);
        //通過類物件獲取所有的成員方法
        Method[] methods = tclass.getDeclaredMethods();
        for (Method method:methods){
            System.out.println("method:"+method);
        }
        //通過類物件獲取getAge成員方法
        Method getAge = tclass.getDeclaredMethod("getAge");
        System.out.println("getAge:"+getAge);
        //通過類物件獲取getAge成員方法
        Method setAge = tclass.getDeclaredMethod("setAge", int.class);
        System.out.println("setAge:"+setAge);
    }
}

獲取成員變數程式碼範例:

package com.reflect;
import java.lang.reflect.Field;
public class TestField {
    public static void main(String[] args) throws Exception {
        Class pcla=Person.class;
        /*//獲取公共存取許可權的成員變數
        Field[] fields = pcla.getFields();
        for (Field field:fields){
            System.out.println("getFild:"+field);
        }
        System.out.println();
        //獲取所有存取許可權的成員變數
        Field[] fielddes = pcla.getDeclaredFields();
        for (Field field:fielddes){
            System.out.println("field:"+field);
        }*/
        Field name = pcla.getDeclaredField("name");
        System.out.println(name);
        Person person=new Person();
        //暴力反射:獲取任意存取許可權修飾符的安全檢查
        name.setAccessible(true);
        //獲取公共成員變數的值
        Object value = name.get(person);
        System.out.println(value);
        //獲取任意存取許可權的成員變數的值
        Object value2 = name.get(person);
        System.out.println("value2:"+value2);
        //設定任意存取許可權的成員變數的值
        name.set(person,"張三");
        Object value3=name.get(person);
        System.out.println("name:"+value3);
    }
}

如何獲取私有變數的值

//暴力反射:獲取任意存取許可權修飾符的安全檢查
name.setAccessible(true);

根據有無主方法判斷程序和執行緒

程序:含有自己的主方法,可以依託自己的主方法啟動,叫做程序

執行緒:沒有自己的主方法,需要依賴其他工具來執行

例如:servlet就需要藉助tomcate來執行,tomcate有自己的一個主方法

反射出現的背景(記住)

舉例:在servlet通過藉助工具tomcate來執行時,tomacate執行專案時存取不到類的資源,由此產生了反射

tomcate為什麼拿不到new的物件

詳解:tomcate不可能通過new的方式來呼叫,因為tomacate是先產生寫好的,類是後來寫的,所以tomcate不知道new的物件的是什麼,可以通過包掃描的方式來獲取檔案路徑,但是這樣也無法使用new的方式,由此產生了反射。

ate有自己的一個主方法

反射出現的背景

舉例:在servlet通過藉助工具tomcate來執行時,tomacate執行專案時存取不到類的資源,由此產生了反射

tomcate為什麼拿不到new的物件?

詳解:tomcate不可能通過new的方式來呼叫,因為tomacate是先產生寫好的,類是後來寫的,所以tomcate不知道new的物件的是什麼,可以通過包掃描的方式來獲取檔案路徑,但是這樣也無法使用new的方式,由此產生了反射。

tomcate想呼叫doGet,doPost的方法時,因為這兩個方法不是靜態的,必須通過new物件才能呼叫,但是tomcate又不能建立物件,所以由此產生反射來獲取檔案

到此這篇關於Java反射機制詳解的文章就介紹到這了,更多相關Java反射內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com