Java反射機制詳解

類的聲明周期

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反射內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: