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!