Java反射機制詳解(一)

接下來我們將介紹Java反射機制的一系列的知識。本篇文章主要針對Java反射機制的介紹以及反射API的使用知識。

一、概述

Java反射機制定義

Java反射機制是在運行狀態中,對于任意一個類,都能夠知道這個類中的所有屬性和方法;對于任意一個對象,都能夠調用它的任意一個方法和屬性;這種動態獲取的信息以及動態調用對象的方法的功能稱為java語言的反射機制。
Java 反射機制的功能

1.在運行時判斷任意一個對象所屬的類。

2.在運行時構造任意一個類的對象。

3.在運行時判斷任意一個類所具有的成員變量和方法。

4.在運行時調用任意一個對象的方法。

5.生成動態代理。

Java 反射機制的應用場景

1.逆向代碼 ,例如反編譯

2.與注解相結合的框架 例如Retrofit

3.單純的反射機制應用框架 例如EventBus

4.動態生成類框架 例如Gson

二、通過Java反射查看類信息

獲得Class對象
每個類被加載之后,系統就會為該類生成一個對應的Class對象。通過該Class對象就可以訪問到JVM中的這個類。

在Java程序中獲得Class對象通常有如下三種方式:

1.使用Class類的forName(String clazzName)靜態方法。該方法需要傳入字符串參數,該字符串參數的值是某個類的全限定名(必須添加完整包名)。

2.調用某個類的class屬性來獲取該類對應的Class對象。

3.調用某個對象的getClass()方法。該方法是java.lang.Object類中的一個方法。

//第一種方式 通過Class類的靜態方法——forName()來實現
class1 = Class.forName("com.lvr.reflection.Person");
//第二種方式 通過類的class屬性
class1 = Person.class;
//第三種方式 通過對象getClass方法
Person person = new Person();
Class<?> class1 = person.getClass();

獲取class對象的屬性、方法、構造函數等

1.獲取class對象的成員變量

Field[] allFields = class1.getDeclaredFields();//獲取class對象的所有屬性
Field[] publicFields = class1.getFields();//獲取class對象的public屬性
Field ageField = class1.getDeclaredField("age");//獲取class指定屬性
Field desField = class1.getField("des");//獲取class指定的public屬性

2.獲取class對象的方法

Method[] methods = class1.getDeclaredMethods();//獲取class對象的所有聲明方法
Method[] allMethods = class1.getMethods();//獲取class對象的所有public方法 包括父類的方法
Method method = class1.getMethod("info", String.class);//返回次Class對象對應類的、帶指定形參列表的public方法
Method declaredMethod = class1.getDeclaredMethod("info", String.class);//返回次Class對象對應類的、帶指定形參列表的方法

3.獲取class對象的構造函數

Constructor<?>[] allConstructors = class1.getDeclaredConstructors();//獲取class對象的所有聲明構造函數
Constructor<?>[] publicConstructors = class1.getConstructors();//獲取class對象public構造函數
Constructor<?> constructor = class1.getDeclaredConstructor(String.class);//獲取指定聲明構造函數
Constructor publicConstructor = class1.getConstructor(String.class);//獲取指定聲明的public構造函數

4.其他方法

Annotation[] annotations = (Annotation[]) class1.getAnnotations();//獲取class對象的所有注解
Annotation annotation = (Annotation) class1.getAnnotation(Deprecated.class);//獲取class對象指定注解
Type genericSuperclass = class1.getGenericSuperclass();//獲取class對象的直接超類的 Type
Type[] interfaceTypes = class1.getGenericInterfaces();//獲取class對象的所有接口的type集合

獲取class對象的信息
比較多。

boolean isPrimitive = class1.isPrimitive();//判斷是否是基礎類型
boolean isArray = class1.isArray();//判斷是否是集合類
boolean isAnnotation = class1.isAnnotation();//判斷是否是注解類
boolean isInterface = class1.isInterface();//判斷是否是接口類
boolean isEnum = class1.isEnum();//判斷是否是枚舉類
boolean isAnonymousClass = class1.isAnonymousClass();//判斷是否是匿名內部類
boolean isAnnotationPresent = class1.isAnnotationPresent(Deprecated.class);//判斷是否被某個注解類修飾
String className = class1.getName();//獲取class名字 包含包名路徑
Package aPackage = class1.getPackage();//獲取class的包信息
String simpleName = class1.getSimpleName();//獲取class類名
int modifiers = class1.getModifiers();//獲取class訪問權限
Class<?>[] declaredClasses = class1.getDeclaredClasses();//內部類
Class<?> declaringClass = class1.getDeclaringClass();//外部類

三、通過Java反射生成并操作對象

生成類的實例對象

1.使用Class對象的newInstance()方法來創建該Class對象對應類的實例。這種方式要求該Class對象的對應類有默認構造器,而執行newInstance()方法時實際上是利用默認構造器來創建該類的實例。

2.先使用Class對象獲取指定的Constructor對象,再調用Constructor對象的newInstance()方法來創建該Class對象對應類的實例。通過這種方式可以選擇使用指定的構造器來創建實例。

//第一種方式 Class對象調用newInstance()方法生成
Object obj = class1.newInstance();
//第二種方式 對象獲得對應的Constructor對象,再通過該Constructor對象的newInstance()方法生成
Constructor<?> constructor = class1.getDeclaredConstructor(String.class);//獲取指定聲明構造函數
obj = constructor.newInstance("hello");

調用類的方法

1.通過Class對象的getMethods()方法或者getMethod()方法獲得指定方法,返回Method數組或對象。

2.調用Method對象中的Object invoke(Object obj, Object... args)方法。第一個參數對應調用該方法的實例對象,第二個參數對應該方法的參數。

 // 生成新的對象:用newInstance()方法
 Object obj = class1.newInstance();
//首先需要獲得與該方法對應的Method對象
Method method = class1.getDeclaredMethod("setAge", int.class);
//調用指定的函數并傳遞參數
method.invoke(obj, 28);

當通過Method的invoke()方法來調用對應的方法時,Java會要求程序必須有調用該方法的權限。如果程序確實需要調用某個對象的private方法,則可以先調用Method對象的如下方法。
setAccessible(boolean flag):將Method對象的acessible設置為指定的布爾值。值為true,指示該Method在使用時應該取消Java語言的訪問權限檢查;值為false,則知識該Method在使用時要實施Java語言的訪問權限檢查。

訪問成員變量值

1.通過Class對象的getFields()方法或者getField()方法獲得指定方法,返回Field數組或對象。

2.Field提供了兩組方法來讀取或設置成員變量的值:
getXXX(Object obj):獲取obj對象的該成員變量的值。此處的XXX對應8種基本類型。如果該成員變量的類型是引用類型,則取消get后面的XXX。
setXXX(Object obj,XXX val):將obj對象的該成員變量設置成val值。

//生成新的對象:用newInstance()方法 
Object obj = class1.newInstance();
//獲取age成員變量
Field field = class1.getField("age");
//將obj對象的age的值設置為10
field.setInt(obj, 10);
//獲取obj對象的age的值
field.getInt(obj);

還有很多其他操作,就不一一介紹了。

下篇文章我們將介紹反射和動態代理的相關知識。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容

  • 1. Java基礎部分 基礎部分的順序:基本語法,類相關的語法,內部類的語法,繼承相關的語法,異常的語法,線程的語...
    子非魚_t_閱讀 31,764評論 18 399
  • 一、概述 1、Java反射機制(Java-Reflect): 在運行狀態中,對于任意一個類,都能夠知道這個類中的所...
    年少懵懂丶流年夢閱讀 4,472評論 0 5
  • Spring Cloud為開發人員提供了快速構建分布式系統中一些常見模式的工具(例如配置管理,服務發現,斷路器,智...
    卡卡羅2017閱讀 134,908評論 18 139
  • No .1 《解憂雜貨店》 來信放進卷簾門前的瓦楞紙箱,第二天回信變出現在后門的牛奶箱所有糾結做選擇的人心里早就有...
    腦子放假閱讀 176評論 0 0
  • 記錄一下吧! 我的奶奶…… 她是個接生婆!煙抽得很兇!記憶中她多數說話身邊總是煙霧繚繞的。 她待我堂哥、我弟較好,...
    金豬欲孽閱讀 362評論 0 0