摘要:反射學習的靈魂我們從最初的,到面向對象部分,我們可以將代碼在計算機中經歷的階段分為三部分源代碼階段類對象階段運行時階段而我們知道,中一個類在源代碼階段,是存儲在硬盤中的,而編譯后,就已經被加載到內存中區(qū),那么有沒有一種方法可以在這種情況下
反射:Web學習的靈魂
我們從最初的 javac -HelloWorld.java,到面向對象部分,我們可以將Java代碼在計算機中經歷的階段分為三部分:Scource源代碼階段 —— Class類對象階段 —— Runtime運行時階段 而我們知道,Java中一個類在源代碼階段,是存儲在硬盤中的,而編譯后,就已經被加載到內存中區(qū),那么有沒有一種方法可以在這種情況下,獲取或者修改它的方法或者屬性呢?這就是我們今天所講的Java反射機制(一) 概述以及好處 (1) 概述
JAVA反射機制是在運行狀態(tài)中,對于任意一個類,都能夠知道這個類的所有屬性和方法;對于任意一個對象,都能夠調用它的任意一個方法和屬性;這種動態(tài)獲取的信息以及動態(tài)調用對象的方法的功能稱為java語言的反射機制。——百度百科
我們將類分為三部分,成員變量,構造方法,成員方法,代碼編譯后,變成了字節(jié)碼文件(.class文件)而萬物皆對象,所以在字節(jié)碼文件中,又將這三部分分別整合成對象,所以我們得出結論:
反射:將類的各個組成部分封裝成對象,并通過這個Class類型的對象,去使用該文件中的成員變量,構造方法,成員方法
(2) 好處可以在程序“運行時”,對 .class文件進行操作,并且由此獲取以及操作 類中的各個部分
可以解耦,提高程序的擴展性
增加程序的靈活性 (最后案例中體現)
(二) 獲取Class對象的方式(1)將字節(jié)碼文件加載進內存,返回Class對象
多用于配置文件,將類名定義在配置文件中。讀取文件,加載類
Class.forName("全類名")
(2)多用于參數的傳遞
類名.class
(3)getClass()方法在Object類中定義著
多用于對象的獲取字節(jié)碼的方式
對象.getClass():
三種方式的代碼實現
package cn.ideal.reflect; import cn.ideal.domain.Student; public class ReflectDemo1 { public static void main(String[] args) throws Exception { //1.Class.forName("全類名") Class cls1 = Class.forName("cn.ideal.domain.Student"); System.out.println(cls1); //2.類名.class Class cls2 = Student.class; System.out.println(cls2); //3.對象.getClass() Student p = new Student(); Class cls3 = p.getClass(); System.out.println(cls3); //用 == 比較 System.out.println(cls1 == cls2);//true System.out.println(cls2 == cls3);//true } } //運行結果 class cn.ideal.domain.Student class cn.ideal.domain.Student class cn.ideal.domain.Student true true
通過上面的案例我們可以得出:
同一個字節(jié)碼文件(*.class)在一次程序運行過程中,只加載一次,不論通過哪一種方式獲取的Class對象都是同一個
(三) Class對象功能 (1) 獲取功能:獲取成員變量們
//獲取所有public修飾的成員變量 Field[] getFields() //獲取指定名稱的public修飾的成員變量 Field getField(String name) //獲取所有的成員變量,不考慮修飾符 Field[] getDeclaredFields() Field getDeclaredField(String name)
構造方法們
Constructor>[] getConstructors() ConstructorgetConstructor(類>... parameterTypes) Constructor getDeclaredConstructor(類>... parameterTypes) Constructor>[] getDeclaredConstructors()
獲取成員方法們
Method[] getMethods() Method getMethod(String name, 類>... parameterTypes) Method[] getDeclaredMethods() Method getDeclaredMethod(String name, 類>... parameterTypes)
獲取全類名
String getName()
Field:成員變量
操作: //設置值 void set(Object obj, Object value) //獲取值 get(Object obj) //忽略訪問權限修飾符的安全檢查 setAccessible(true):暴力反射
Constructor:構造方法
創(chuàng)建對象: T newInstance(Object... initargs) 如果使用空參數構造方法創(chuàng)建對象,操作可以簡化:Class對象的newInstance方法
Method:方法對象
//執(zhí)行方法 Object invoke(Object obj, Object... args) //獲取方法名稱 String getName:獲取方法名
我們來具體使用一下上面的一些方法
package cn.ideal.reflect; import cn.ideal.domain.Student; import java.lang.reflect.Field; public class ReflectDemo2 { public static void main(String[] args) throws Exception { //獲取Student的Class對象 Class studentClass = Student.class; /* 1. 獲取成員變量們 * Field[] getFields() * Field getField(String name) * Field[] getDeclaredFields() Field getDeclaredField(String name) */ //Field[] getFields() 獲取所有public修飾的成員變量 Field[] fields = studentClass.getFields(); for (Field field : fields) { System.out.println(field); } System.out.println("----------"); //Field getField(String name) Field a = studentClass.getField("a"); //獲取成員變量a的值 Student s = new Student(); Object value = a.get(s); System.out.println(value); //設置a的值 a.set(s, "張三"); System.out.println(s); System.out.println("----------"); //Field[] getDeclaredFields() 獲取所有的成員變量 Field[] declaredFields = studentClass.getDeclaredFields(); for (Field declaredField : declaredFields) { System.out.println(declaredField); } System.out.println("----------"); //Field getDeclaredField(String name) Field d = studentClass.getDeclaredField("d"); //忽略訪問權限修飾符 d.setAccessible(true);//暴力反射 Object value2 = d.get(s); System.out.println(value2); } } //運行結果 public java.lang.String cn.ideal.domain.Student.a ---------- null Student{name="null", age=0, a="張三", b="null", c="null", d="null"} ---------- private java.lang.String cn.ideal.domain.Student.name private int cn.ideal.domain.Student.age public java.lang.String cn.ideal.domain.Student.a protected java.lang.String cn.ideal.domain.Student.b java.lang.String cn.ideal.domain.Student.c private java.lang.String cn.ideal.domain.Student.d ---------- null
package cn.ideal.reflect; import cn.ideal.domain.Student; import java.lang.reflect.Constructor; public class ReflectDemo3 { public static void main(String[] args) throws Exception { //獲取Student的Class對象 Class studentClass = Student.class; /* 2. 獲取構造方法們 * Constructor>[] getConstructors() * ConstructorgetConstructor(類>... parameterTypes) * Constructor getDeclaredConstructor(類>... parameterTypes) * Constructor>[] getDeclaredConstructors() */ //Constructor getConstructor(類>... parameterTypes) Constructor constructor = studentClass.getConstructor(String.class, int.class); System.out.println(constructor); //創(chuàng)建對象 帶參 Object student = constructor.newInstance("張三", 20); System.out.println(student); System.out.println("----------"); Constructor constructor1 = studentClass.getConstructor(); System.out.println(constructor1); //創(chuàng)建對象 不帶參 Object student1 = constructor1.newInstance(); System.out.println(student1); //創(chuàng)建對象 不帶參 (推薦方法) Object o = studentClass.newInstance(); System.out.println(o); } }
package cn.ideal.reflect; import cn.ideal.domain.Student; import java.lang.reflect.Method; public class ReflectDemo4 { public static void main(String[] args) throws Exception { //獲取Student的Class對象 Class studentClass = Student.class; /* 3. 獲取成員方法們: * Method[] getMethods() * Method getMethod(String name, 類>... parameterTypes) * Method[] getDeclaredMethods() * Method getDeclaredMethod(String name, 類>... parameterTypes) */ //獲取指定名稱的方法 Method study_method = studentClass.getMethod("study"); Student s = new Student(); //執(zhí)行方法 study_method.invoke(s); Method study_method2 = studentClass.getMethod("study", String.class); //執(zhí)行方法 study_method2.invoke(s, "英語"); System.out.println("----------"); //獲取所有public修飾的方法 Method[] methods = studentClass.getMethods(); for (Method method : methods) { System.out.println(method); String name = method.getName(); System.out.println(name); } //獲取類名 String className = studentClass.getName(); System.out.println(className); } }(四) 反射案例
通過修改配置文件,達到不改變該類的任何代碼,可以創(chuàng)建任意類的對象,可以執(zhí)行任意方法,避免了每一次都修改類文件的缺點,同時提高了程序的擴展性
* 實現: 1. 配置文件 2. 反射 * 步驟: 1. 將需要創(chuàng)建的對象的全類名和需要執(zhí)行的方法定義在配置文件中 2. 在程序中加載讀取配置文件 3. 使用反射技術來加載類文件進內存 4. 創(chuàng)建對象 5. 執(zhí)行方法
//pro.properties 自定義配置文件 className=cn.ideal.domain.Student methodName=study
package cn.ideal.reflect; import java.io.InputStream; import java.lang.reflect.Method; import java.util.Properties; public class ReflectTest { public static void main(String[] args) throws Exception { /* 不改變該類的任何代碼,可以創(chuàng)建任意類的對象,可以執(zhí)行任意方法 */ //1.加載配置文件 //創(chuàng)建Properties對象 Properties pro = new Properties(); //加載配置文件,轉換為一個集合 //獲取class目錄下的配置文件 ClassLoader classLoader = ReflectTest.class.getClassLoader(); InputStream is = classLoader.getResourceAsStream("pro.properties"); pro.load(is); //2.獲取配置文件中定義的數據 String className = pro.getProperty("className"); String methodName = pro.getProperty("methodName"); //3.加載該類進內存 Class cls = Class.forName(className); //4.創(chuàng)建對象 Object obj = cls.newInstance(); //5.獲取方法對象 Method method = cls.getMethod(methodName); //6.執(zhí)行方法 method.invoke(obj); }
我僅僅淺薄的談了一下反射的基本知識,一些比較深入的理解由于篇幅問題,放在后面專篇講解,感謝支持。確實有很多不足之處,也希望能與大家交流。
結尾:如果內容中有什么不足,或者錯誤的地方,歡迎大家給我留言提出意見, 蟹蟹大家 !^_^
如果能幫到你的話,那就來關注我吧!(系列文章均會在公眾號第一時間更新)
在這里的我們素不相識,卻都在為了自己的夢而努力 ?一個堅持推送原創(chuàng)Java技術的公眾號:理想二旬不止
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規(guī)行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://systransis.cn/yun/75581.html
摘要:與都繼承自類,在中也是使用字符數組保存字符串,,這兩種對象都是可變的。采用字節(jié)碼的好處語言通過字節(jié)碼的方式,在一定程度上解決了傳統解釋型語言執(zhí)行效率低的問題,同時又保留了解釋型語言可移植的特點。 String和StringBuffer、StringBuilder的區(qū)別是什么?String為什么是不可變的? String和StringBuffer、StringBuilder的區(qū)別 可變性...
摘要:經過前面幾次反射機制的學習,這次用反射的知識寫一個類似于框架處理機制的小。和在引入反射知識前,先簡單介紹下框架和。再次使用反射獲取的最新值,組成返回,同時根據方法的返回值,去中獲取對應的。最后根據的返回值和的名稱組成最終展示的視圖。 經過前面幾次反射機制的學習,這次用反射的知識寫一個類似于Struts框架處理機制的小demo。 Servlet 和 Sturts 在引入反射知識前,先簡單...
摘要:本文是作者自己對中線程的狀態(tài)線程間協作相關使用的理解與總結,不對之處,望指出,共勉。當中的的數目而不是已占用的位置數大于集合番一文通版集合番一文通版垃圾回收機制講得很透徹,深入淺出。 一小時搞明白自定義注解 Annotation(注解)就是 Java 提供了一種元程序中的元素關聯任何信息和著任何元數據(metadata)的途徑和方法。Annotion(注解) 是一個接口,程序可以通過...
摘要:的在日常開發(fā),特別是開發(fā)中使用廣泛,各種框架,測試框架多多少少都會引入一些注解。的功能是作用于程序元數據的特殊類型。而在編譯期使用注解則需要特殊的工具,本文不討論??梢钥吹酵ㄟ^類中的反射方法獲取到了這個類的注解以及其方法的注解。 Java的Annotation在日常開發(fā),特別是java web開發(fā)中使用廣泛,各種web框架,測試框架多多少少都會引入一些注解。若對java注解有一個全面深...
摘要:以實現自己熟悉的東西為導向比如我們做后端開發(fā),首先是常用的循環(huán)迭代條件判斷增刪改成。它是由實現的,不保證元素的順序,也就是說所說元素插入的順序與輸出的順序不一致。 下面是我直播的文字版,直播地址:https://segmentfault.com/l/15...代碼:https://github.com/zhoumengka...整個項目我們我又細分了6個版本來演進,希望更加便于大家對比...
閱讀 2240·2023-04-26 01:57
閱讀 3266·2023-04-25 16:30
閱讀 2338·2021-11-17 09:38
閱讀 1090·2021-10-08 10:14
閱讀 1395·2021-09-23 11:21
閱讀 3693·2019-08-29 17:28
閱讀 3465·2019-08-29 15:27
閱讀 955·2019-08-29 13:04