Java基礎篇之反射機制詳解
思考:在講反射之前,先思考一個問題,java中如何創建一個對象,有哪幾種方式?
Java中創建對象大概有這幾種方式:
- 1、使用new關鍵字:這是我們最常見的也是最簡單的創建對象的方式
- 2、使用Clone的方法:無論何時我們調用一個對象的clone方法,JVM就會創建一個新的對象,將前面的對象的內容全部拷貝進去
- 3、使用反序列化:當我們序列化和反序列化一個對象,JVM會給我們創建一個單獨的對象
上邊是Java中常見的創建對象的三種方式,其實除瞭上邊的三種還有另外一種方式,就是接下來我們要討論的 “反射”
1、反射概述
1.1什麼是反射
反射就是把Java類中的各個部分,映射成一個個的Java對象,拿到這些對象後可以做一些事情。
既然說反射是反射Java類中的各個組成部分,所以說咱們得知道一個類中有哪兒些部分?
例如,一個類有:成員變量,方法,構造方法,等信息,利用反射技術咱們可以把這些組成部分映射成一個個對象。
1.2、反射能幹什麼
說完反射的概念後,咱們說一下反射能幹什麼?
一般來說反射是用來做框架的,或者說可以做一些抽象度比較高的底層代碼,反射在日常的開發中用到的不多,但是咱們還必須搞懂它,因為搞懂瞭反射以後,可以幫助咱們理解框架的一些原理。所以說有一句很經典的話:反射是框架設計的靈魂。現在說完這個可能還不太能理解,不急,等下說完一個快速入門的例子後,應該會稍微有點感覺
1.3、怎麼得到想反射的類
剛才已經說過,反射是對一個類進行解剖,想解剖一個東西,前提是首先你得拿到這個東西,那麼怎麼得到咱們想解剖的類呢?
首先大傢要明白一點,咱們寫的代碼是存儲在後綴名是 .java的文件裡的,但是它會被編譯,最終真正去執行的是編譯後的 .class文件。Java是面向對象的語言,一切皆對象,所以java認為 這些編譯後的 class文件,這種事物也是一種對象,它也給抽象成瞭一種類,這個類就是Class,大傢可以去AIP裡看一下這個類
所以拿到這個類後,就相當於拿到瞭咱們想解剖的類,那怎麼拿到這個類?
看API文檔後,有一個方法forName(String className); 而且是一個靜態的方法,這樣咱們就可以得到想反射的類瞭
到這裡,看Class clazz = Class.forName("com.cj.test.Person");
這個應該有點感覺瞭吧
Class.forName("com.cj.test.Person");
因為這個方法裡接收的是個字符串,字符串的話,我們就可以寫在配置文件裡,然後利用反射生成我們需要的對象,這才是我們想要的。很多框架裡都有類似的配置
2、解剖類
我們知道一個類裡一般有構造函數、方法、成員變量(字段/屬性)這三部分組成
翻閱API文檔,可以看到
Class對象提供瞭如下常用方法:
- public Constructor getConstructor(Class<?>…parameterTypes)
- public Method getMethod(String name,Class<?>… parameterTypes)
- public Field getField(String name)
- public Constructor getDeclaredConstructor(Class<?>…parameterTypes)
- public Method getDeclaredMethod(String name,Class<?>… parameterTypes)
- public Field getDeclaredField(String name)
這些方法分別用於幫咱們從類中解剖出構造函數、方法和成員變量(屬性)。
然後把解剖出來的部分,分別用Constructor、Method、Field對象表示。
2.1反射構造方法
2.1.1反射無參的構造函數
可以看到 默認的無參構造方法執行瞭
從上邊的例子看出,要想反射,首先第一步就是得到類的字節碼
所以簡單說一下得到類的字節碼的幾種方式
- (1)、Class.forName("com.cj.test.Person"); 這就是上邊我們用的方式
- (2)、對象.getClass();
- (3)、類名.class;
2.1.2反射“一個參數”的構造函數
2.1.3反射“多個參數”的構造函數
2.1.4反射“私有”的構造函數
註意:在反射私有的構造函數時,用普通的clazz.getConstructor()會報錯,因為它是私有的,所以提供瞭專門反射私有構造函數的方法 clazz.getDeclaredConstructor(int.class);//讀取私有的構造函數
,用這個方法讀取完還需要設置一下暴力反射才可以
c.setAccessible(true);//暴力反射
2.1.5反射得到類中所有的構造函數
2.2反射類中的方法
package com.cj.test; import java.util.Date; public class Person { public Person(){ System.out.println("默認的無參構造方法執行瞭"); } public Person(String name){ System.out.println("姓名:"+name); } public Person(String name,int age){ System.out.println(name+"="+age); } private Person(int age){ System.out.println("年齡:"+age); } public void m1() { System.out.println("m1"); } public void m2(String name) { System.out.println(name); } public String m3(String name,int age) { System.out.println(name+":"+age); return "aaa"; } private void m4(Date d) { System.out.println(d); } public static void m5() { System.out.println("m5"); } public static void m6(String[] strs) { System.out.println(strs.length); } public static void main(String[] args) { System.out.println("main"); } }
package com.cj.test; import java.lang.reflect.Method; import java.util.Date; import org.junit.Test; public class Demo2 { @Test//public void m1() public void test1() throws Exception{ Class clazz = Class.forName("com.cj.test.Person"); Person p = (Person)clazz.newInstance(); Method m = clazz.getMethod("m1", null); m.invoke(p, null); } @Test//public void m2(String name) public void test2() throws Exception{ Class clazz = Person.class; Person p = (Person) clazz.newInstance(); Method m = clazz.getMethod("m2", String.class); m.invoke(p, "張三"); } @Test//public String m3(String name,int age) public void test3() throws Exception{ Class clazz = Person.class; Person p = (Person) clazz.newInstance(); Method m = clazz.getMethod("m3", String.class,int.class); String returnValue = (String)m.invoke(p, "張三",23); System.out.println(returnValue); } @Test//private void m4(Date d) public void test4() throws Exception{ Class clazz = Person.class; Person p = (Person) clazz.newInstance(); Method m = clazz.getDeclaredMethod("m4", Date.class); m.setAccessible(true); m.invoke(p,new Date()); } @Test//public static void m5() public void test5() throws Exception{ Class clazz = Person.class; Method m = clazz.getMethod("m5", null); m.invoke(null,null); } @Test//private static void m6(String[] strs) public void test6() throws Exception{ Class clazz = Person.class; Method m = clazz.getDeclaredMethod("m6",String[].class); m.setAccessible(true); m.invoke(null,(Object)new String[]{"a","b"}); } @Test public void test7() throws Exception{ Class clazz = Person.class; Method m = clazz.getMethod("main",String[].class); m.invoke(null,new Object[]{new String[]{"a","b"}}); } }
*****註意:看下上邊代碼裡test6和test7的invoke方法裡傳的參數和其他的有點不一樣
這是因為 jdk1.4和jdk1.5處理invoke方法有區別
1.5:public Object invoke(Object obj,Object…args)
1.4:public Object invoke(Object obj,Object[] args)
由於JDK1.4和1.5對invoke方法的處理有區別, 所以在反射類似於main(String[] args) 這種參數是數組的方法時需要特殊處理
啟動Java程序的main方法的參數是一個字符串數組,即public static void main(String[] args),通過反射方式來調用這個main方法時,如何為invoke方法傳遞參數呢?按jdk1.5的語法,整個數組是一個參數,而按jdk1.4的語法,數組中的每個元素對應一個參數,當把一個字符串數組作為參數傳遞給invoke方法時,javac會到底按照哪種語法進行處理呢?jdk1.5肯定要兼容jdk1.4的語法,會按jdk1.4的語法進行處理
,即把數組打散成為若幹個單獨的參數。所以,在給main方法傳遞參數時,不能使用代碼mainMethod.invoke(null,new String[]{“xxx”}),javac隻把它當作jdk1.4的語法進行理解,而不把它當作jdk1.5的語法解釋,因此會出現參數個數不對的問題。
上述問題的解決方法:
- (1)mainMethod.invoke(null,new Object[]{new String[]{"xxx"}});
這種方式,由於你傳的是一個數組的參數,所以為瞭向下兼容1.4的語法,javac遇到數組會給你拆開成多個參數,但是由於咱們這個Object[ ] 數組裡隻有一個元素值,所以就算它拆也沒關系
- (2)mainMethod.invoke(null,(Object)new String[]{"xxx"});
這種方式相當於你傳的參數是一個對象,而不是數組,所以就算是按照1.4的語法它也不會拆,所以問題搞定
編譯器會作特殊處理,編譯時不把參數當作數組看待,也就不會數組打散成若幹個參數瞭
對上邊的描述進行一下總結:在反射方法時,如果方法的參數是一個數組,考慮到向下兼容問題,會按照JDK1.4的語法來對待(JVM會把傳遞的數組參數拆開,拆開就會報參數的個數不匹配的錯誤)
解決辦法:防止JVM拆開你的數組
- 方式一:把數組看做是一個Object對象
- 方式二:重新構建一個Object數組,那個參數數組作為唯一的元素存在。
2.3反射類中的屬性字段
package com.cj.test; import java.util.Date; public class Person { public String name="李四"; private int age = 18; public static Date time; public int getAge() { return age; } public Person(){ System.out.println("默認的無參構造方法執行瞭"); } public Person(String name){ System.out.println("姓名:"+name); } public Person(String name,int age){ System.out.println(name+"="+age); } private Person(int age){ System.out.println("年齡:"+age); } public void m1() { System.out.println("m1"); } public void m2(String name) { System.out.println(name); } public String m3(String name,int age) { System.out.println(name+":"+age); return "aaa"; } private void m4(Date d) { System.out.println(d); } public static void m5() { System.out.println("m5"); } public static void m6(String[] strs) { System.out.println(strs.length); } public static void main(String[] args) { System.out.println("main"); } }
package com.cj.test; import java.lang.reflect.Field; import java.util.Date; import org.junit.Test; public class Demo3 { //public String name="李四"; @Test public void test1() throws Exception{ Class clazz = Person.class; Person p = (Person)clazz.newInstance(); Field f = clazz.getField("name"); String s = (String)f.get(p); System.out.println(s); //更改name的值 f.set(p, "王六"); System.out.println(p.name); } @Test//private int age = 18; public void test2() throws Exception{ Class clazz = Person.class; Person p = (Person)clazz.newInstance(); Field f = clazz.getDeclaredField("age"); f.setAccessible(true); int age = (Integer)f.get(p); System.out.println(age); f.set(p, 28); age = (Integer)f.get(p); System.out.println(age); } @Test//public static Date time; public void test3() throws Exception{ Class clazz = Person.class; Field f = clazz.getField("time"); f.set(null, new Date()); System.out.println(Person.time); } }
以上就是自己對Java中反射的一些學習總結,歡迎大傢留言一起學習、討論
看完上邊有關反射的東西, 對常用框架裡的配置文件是不是有點思路瞭
上邊是Spring配置文件裡的常見的bean配置,這看起來是不是可以用反射很輕易的就可以實現:解析xml然後把xml裡的內容作為參數,利用反射創建對象。
以上所述是小編給大傢介紹的Java基礎篇之反射機制詳解,希望對大傢有所幫助。在此也非常感謝大傢對WalkonNet網站的支持!