Java反射機(jī)制是在運(yùn)行狀態(tài)中,對(duì)于任意一個(gè)類,都能夠知道這個(gè)類的所有屬性和方法;對(duì)于任意一個(gè)對(duì)象,都能夠調(diào)用它的任意一個(gè)方法和屬性;這種動(dòng)態(tài)獲取的信息以及動(dòng)態(tài)調(diào)用對(duì)象的方法的功能稱為java語(yǔ)言的反射機(jī)制。
反射就是把java類中的各種成分映射成一個(gè)個(gè)的Java對(duì)象。
例如:一個(gè)類有:成員變量、方法、構(gòu)造方法、包等等信息,利用反射技術(shù)可以對(duì)一個(gè)類進(jìn)行解剖,把個(gè)個(gè)組成部分映射成一個(gè)個(gè)對(duì)象。(其實(shí):一個(gè)類中這些成員方法、構(gòu)造方法、在加入類中都有一個(gè)類來(lái)描述)
反射
Q:調(diào)用類對(duì)象.class 和 forName(類名)的區(qū)別?
Class《A》 classA = A.class;Class《A》 classA = Class.forName(“A”);
A:僅使用.class不能進(jìn)行第一次靜態(tài)初始化, forname函數(shù)則可以例如B是A的基類,下面這段代碼如何?假設(shè)有父子2個(gè)類,如下:
static class Parent { }
static class Son extends Parent{}
Q:用instanceof 可以和父類比較嗎,且會(huì)返回true嗎?
Son son = new Son(); if (son instanceof Parent) { System.out.println(“a instanof B”); }
A:可以比較,且返回true。
Q:用getClass并用== 可以和父類比較嗎,且會(huì)返回true嗎,下面這樣:注意A是B的子類。
Son son = new Son(); if (son.getClass() == Parent.class){ System.out.println(“son class == Parent.class”); }
A:不可以,編譯就會(huì)報(bào)錯(cuò)了。和Class《泛型》的 ==號(hào)比較有關(guān)。

因?yàn)間etClass返回的是《? extends Son》, .class返回的是Class《Parent》
Q:用getClass并用.equals可以和父類比較嗎,且會(huì)返回true嗎,下面這樣:
Son son = new Son(); if (son.getClass().equals(Parent.class)){ System.out.println(“son class.equals(Parent.class)”); }
A:可以比較,正常編譯, 但是會(huì)返回false,即不相等!
Q:getDeclaredXXX 有哪幾種?A:5種:
注解Annotation
內(nèi)部類Classed
構(gòu)造方法Construcotor
字段Field
方法Method

Q:getMethods()返回哪些方法, getDeclaredMethods()會(huì)返回哪些方法?
A:getMethods()返回 本類、父類、父接口 的public方法getDeclaredMethods()只 返回本類的 所有 方法其他getXXX和getDeclaredXXX的區(qū)別同理。拿到Filed、Method、Constructor之后咋用
Method可以invoke(object, args)
Constructor可以newInstance(Object…)來(lái)做構(gòu)造調(diào)用。
Filed可以用get(object)、set(object)來(lái)設(shè)置屬性值。
Q:反射拿到Method對(duì)象后, 該對(duì)象.getModifiers() 是干嘛的?
A:返回該方法的修飾符,并且是1個(gè)整數(shù)。

Q:下面這段代碼會(huì)發(fā)生什么?
package com.huawei.test
public class A { public A(int i ) { System.out.printf(“i=” +i); }
public static void main(String[] args) { try { A a = (A)Class.forName(“com.huawei.test.A”).newInstance(); } catch (ClassNotFoundException e) { System.out.printf(“ClassNotFoundException”); } catch (InstantiationException e) { System.out.printf(“InstantiationException”); } catch (IllegalAccessException e) { System.out.printf(“IllegalAccessException”); } }}
A:打印InstantiationException初始化錯(cuò)誤。因?yàn)锳沒(méi)有默認(rèn)構(gòu)造器了,所以不可以用newInstance來(lái)構(gòu)造。應(yīng)該改成這樣,通過(guò)獲取正確的構(gòu)造器來(lái)進(jìn)行構(gòu)造。
A a = (A)Class.forName(“A”).getConstructor(int.class).newInstance(123);
Q:如何提高反射的效率?
A:
使用高性能反射包,例如ReflectASM
緩存反射的對(duì)象,避免每次都要重復(fù)去字節(jié)碼中獲取。(緩存!緩存!)
method反射可設(shè)置method.setAccessible(true)來(lái)關(guān)閉安全檢查。
盡量不要getMethods()后再遍歷篩選,而直接用getMethod(methodName)來(lái)根據(jù)方法名獲取方法
利用hotspot虛擬機(jī)中的反射優(yōu)化技術(shù)(jit技術(shù))
參考資料: https://segmentfault.com/q/1010000003004720https://www.cnblogs.com/coding-night/p/10772631.html
Q:用反射獲取到的method對(duì)象, 是返回一個(gè)method引用,還是返回1個(gè)拷貝的method對(duì)象?
A:反射拿method對(duì)象時(shí), 會(huì)做一次拷貝,而不是直接返回引用,因此最好對(duì)頻繁使用的同一個(gè)method做緩存,而不是每次都去查找。

Q:getMethods()后自己做遍歷獲取方法,和getMethod(methodName) 直接獲取方法, 為什么性能會(huì)有差異?
A:getMethods() 返回method數(shù)組時(shí),每個(gè)method都做了一次拷貝。getMethod(methodName)只會(huì)返回那個(gè)方法的拷貝, 性能的差異就體現(xiàn)在拷貝上。

Q:獲取方法時(shí),jvm內(nèi)部其實(shí)有緩存,但是返回給外部時(shí)依然會(huì)做拷貝。那么該method的緩存是持久存在的嗎?
A:不是持久存在的,內(nèi)存不足時(shí)會(huì)被回收。源碼如下:

private Class.ReflectionData《T》 reflectionData() { SoftReference《Class.ReflectionData《T》》 reflectionData = this.reflectionData; int classRedefinedCount = this.classRedefinedCount; Class.ReflectionData rd; return reflectionData != null && (rd = (Class.ReflectionData)reflectionData.get()) != null && rd.redefinedCount == classRedefinedCount ? rd : this.newReflectionData(reflectionData, classRedefinedCount);}

可以看到這是一個(gè)軟引用。
軟引用的定義:內(nèi)存緊張時(shí)可能會(huì)被回收,不過(guò)也可以通過(guò)-XX:SoftRefLRUPolicyMSPerMB參數(shù)控制回收的時(shí)機(jī),只要發(fā)生GC就會(huì)將其回收。
如果reflectionData被回收之后,又執(zhí)行了反射方法,那只能通過(guò)newReflectionData方法重新創(chuàng)建一個(gè)這樣的對(duì)象了。
Q:反射是線程安全的嗎?
A:是線程安全的。獲取反射的數(shù)據(jù)時(shí),通過(guò)cas去獲取。cas概念可以見(jiàn)多線程一節(jié)。

Q:
a普通方法調(diào)用
b反射方法調(diào)用
c關(guān)閉安全檢查的反射方法調(diào)用,性能差異如下:

b反射方法調(diào)用和c關(guān)閉安全檢查的反射方法調(diào)用的性能差異在哪?普通方法調(diào)用和關(guān)閉安全檢查的反射方法調(diào)用的性能差異在哪?
A:
安全檢查的性能消耗在于
SecurityManager.checkPermission(SecurityConstants.CHECK_MEMBER_ACCESS_PERMISSION);
這項(xiàng)檢測(cè)需要運(yùn)行時(shí)申請(qǐng)
RuntimePermission(“accessDeclaredMembers”)
所以如果不考慮安全檢查, 對(duì)反射方法調(diào)用invoke時(shí), 應(yīng)當(dāng)設(shè)置 Method#setAccessible(true)
普通方法和反射方法的性能差異在于
Method#invoke 方法會(huì)對(duì)參數(shù)做封裝和解封操作
需要檢查方法可見(jiàn)性
需要校驗(yàn)參數(shù)
反射方法難以內(nèi)聯(lián)
JIT 無(wú)法優(yōu)化
編輯:lyn
-
反射
+關(guān)注
關(guān)注
0文章
69瀏覽量
16225 -
JAVA語(yǔ)言
+關(guān)注
關(guān)注
0文章
138瀏覽量
21528
原文標(biāo)題:爆文速遞| Java程序員都要懂得知識(shí)點(diǎn):反射
文章出處:【微信號(hào):Huawei_Developer,微信公眾號(hào):華為開(kāi)發(fā)者社區(qū)】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。
發(fā)布評(píng)論請(qǐng)先 登錄
阻燃系列基礎(chǔ)知識(shí)
無(wú)刷電機(jī)驅(qū)動(dòng)器的基礎(chǔ)知識(shí)
知識(shí)分享-有限上升時(shí)間信號(hào)的反射波形(信號(hào)完整性揭秘)
【「Yocto項(xiàng)目實(shí)戰(zhàn)教程:高效定制嵌入式Linux系統(tǒng)」閱讀體驗(yàn)】01初讀體驗(yàn)
【「# 運(yùn)算放大器參數(shù)解析與LTspice應(yīng)用仿真」閱讀體驗(yàn)】+全書概覽與第一章閱讀分享
C51單片機(jī)及C語(yǔ)言知識(shí)點(diǎn)必備秘籍
電機(jī)選型計(jì)算公式與知識(shí)點(diǎn)匯總
最易讀懂的理工科基礎(chǔ)叢書——圖解電機(jī)基礎(chǔ)知識(shí)入門
射頻基礎(chǔ)知識(shí)培訓(xùn)(華為)—PPT版
電氣工程師必知必會(huì)的100個(gè)電?知識(shí)點(diǎn)分享
開(kāi)關(guān)電源的基礎(chǔ)知識(shí)題目及答案(免積分)
【北京迅為】iTOP-RK3568OpenHarmony系統(tǒng)南向驅(qū)動(dòng)開(kāi)發(fā)GPIO基礎(chǔ)知識(shí)
詳解Java基礎(chǔ)知識(shí)點(diǎn)反射
評(píng)論