java反射
反射:一句話來說就是把JAVA中的每一個成分解析在為相應的JAVA類。對java中的每一個成分都解析成一個類。
1. Class類
Class類代表JAVA類,它的執行個體對應執行個體在記憶體中的位元組碼。每個類被載入到記憶體中,都會被解析成位元組碼。
那麼Class的執行個體就表示這份位元組碼。
得到Class執行個體對象的方法:
1)直接類名調用: Person.class;
2)對象調用: new Person().getClass();
3)從記憶體中載入: Class.forName("java.lang.String");
2.java反射的三個重要類:
1)Constructor類:代表類中的一個構造方法。
Constructor<String> constructor = String.class.getConstructor(String.class);//只取得公有方法。
//擷取構造方法,傳入一個參數,並且參數為String類型的構造方法
String string = (String)constructor.newInstance(new String("abc"));
//通過構造方法反射執行個體化對象。傳入的參數必須是和擷取的構造方法中取得的參數是一樣的。
2)Method:代表類中的一個方法。以下代碼為反射成員方法:
System.out.println("----------------反射成員方法-------------------------");methodCharAt = String.class.getMethod("charAt", int.class);String str1 = "abc";// 下面的invoke則要首先指定是哪個對象去調用上面指定的方法,後面一個則是傳入的參數。System.out.println(methodCharAt.invoke(str1, 1));// methodCharAt.invoke(null,1); //這樣則是表示靜態方法調用。
注意: 而這裡的反射main方法則是需要對main方法中的參數進行封裝。因為jdk1.5為了相容jdk以前的版本。會對傳入的資料進行拆分成單個對象。
System.out.println("----------------反射main方法-------------------------");Method mainMethod = Class.forName("com.test.Person").getMethod("main", String[].class);//System.out.println("main方法反射的錯誤傳參方式。");//mainMethod.invoke(null,new String[]{"ab","abc"});/** * 異常:wrong number of arguments * **//** * jdk1.5為了相容jdk1.4,在傳入一個Object類型的數組(String 直接繼承Object),invoke給自動的進行拆開 * 其中的每一個元素作為一個參數。這樣就會導致了參數個數的不匹配。 * 可能是因為,jdk1.5是支援可變參數的,但是在1.5以前是不支援可變參數的。 * * ***/System.out.println("main方法反射的第一種傳參方式。");//這裡就將數組再次的去打包成一個包,這樣被拆開的時候還一個數組。mainMethod.invoke(null,(Object)new String[]{"ab","abc"});System.out.println("main方法反射的第二種傳參方式。");//這裡強制轉換成一個Object ,目的只是去騙過編譯器,//這樣編譯器不會去進行拆分操作。傳入的依舊是一個數組。mainMethod.invoke(null,new Object[]{new String[]{"ab","abc"}});
3)代表類中一個欄位.
Field fieldX = pt1.getClass().getDeclaredField("x");
如果為私用變數則不能夠進行輸出,那麼就可以去進行暴力反射:
fieldX.setAccessible(true);
3.反射的作用:
1)實現動態代理
2)實現架構