生活随笔
收集整理的這篇文章主要介紹了
什么是反射?反射有什么用处
小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.
什么是反射?反射有什么用處?
1. 什么是反射?
“反射(Reflection)能夠讓運(yùn)行于JVM中的程序檢測(cè)和修改運(yùn)行時(shí)的行為。”這個(gè)概念常常會(huì)和內(nèi)?。↖ntrospection)混淆,以下是這兩個(gè)術(shù)語(yǔ)在Wikipedia中的解釋:
內(nèi)省用于在運(yùn)行時(shí)檢測(cè)某個(gè)對(duì)象的類型和其包含的屬性;
反射用于在運(yùn)行時(shí)檢測(cè)和修改某個(gè)對(duì)象的結(jié)構(gòu)及其行為。
從它們的定義可以看出,內(nèi)省是反射的一個(gè)子集。有些語(yǔ)言支持內(nèi)省,但并不支持反射,如C++。
內(nèi)省示例:instanceof運(yùn)算符用于檢測(cè)某個(gè)對(duì)象是否屬于特定的類。
1
2
3
4
|
if (obj instanceof Dog) {
Dog d = (Dog) obj;
d.bark();
}
|
反射示例:Class.forName()方法可以通過(guò)類或接口的名稱(一個(gè)字符串或完全限定名)來(lái)獲取對(duì)應(yīng)的Class對(duì)象。forName方法會(huì)觸發(fā)類的初始化。
1
2
3
4
5
|
Class<?> c = Class.forName("classpath.and.classname");
Object dog = c.newInstance();
Method m = c.getDeclaredMethod("bark", new Class<?>[0]);
m.invoke(dog);
|
在Java中,反射更接近于內(nèi)省,因?yàn)槟銦o(wú)法改變一個(gè)對(duì)象的結(jié)構(gòu)。雖然一些API可以用來(lái)修改方法和屬性的可見(jiàn)性,但并不能修改結(jié)構(gòu)。
2. 我們?yōu)楹涡枰瓷洌?/h2>
反射能夠讓我們:
在運(yùn)行時(shí)檢測(cè)對(duì)象的類型;
動(dòng)態(tài)構(gòu)造某個(gè)類的對(duì)象;
檢測(cè)類的屬性和方法;
任意調(diào)用對(duì)象的方法;
修改構(gòu)造函數(shù)、方法、屬性的可見(jiàn)性;
以及其他。
反射是框架中常用的方法。
例如,JUnit通過(guò)反射來(lái)遍歷包含 @Test 注解的方法,并在運(yùn)行單元測(cè)試時(shí)調(diào)用它們。(這個(gè)連接中包含了一些JUnit的使用案例)
對(duì)于Web框架,開(kāi)發(fā)人員在配置文件中定義他們對(duì)各種接口和類的實(shí)現(xiàn)。通過(guò)反射機(jī)制,框架能夠快速地動(dòng)態(tài)初始化所需要的類。
例如,Spring框架使用如下的配置文件:
1
2
3
|
<bean id="someID" class="com.programcreek.Foo">
<property name="someField" value="someValue" />
</bean>
|
當(dāng)Spring容器處理<bean>元素時(shí),會(huì)使用Class.forName("com.programcreek.Foo")來(lái)初始化這個(gè)類,并再次使用反射獲取<property>元素對(duì)應(yīng)的setter方法,為對(duì)象的屬性賦值。
Servlet也會(huì)使用相同的機(jī)制:
1
2
3
4
|
<servlet>
<servlet-name>someServlet</servlet-name>
<servlet-class>com.programcreek.WhyReflectionServlet</servlet-class>
<servlet>
|
3. 如何使用反射?
讓我們通過(guò)幾個(gè)典型的案例來(lái)學(xué)習(xí)如何使用反射。
示例1:獲取對(duì)象的類型名稱。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
package myreflection;
import java.lang.reflect.Method;
public class ReflectionHelloWorld {
public static void main(String[] args){
Foo f = new Foo();
System.out.println(f.getClass().getName());
}
}
class Foo {
public void print() {
System.out.println("abc");
}
}
|
輸出:
示例2:調(diào)用未知對(duì)象的方法。
在下列代碼中,設(shè)想對(duì)象的類型是未知的。通過(guò)反射,我們可以判斷它是否包含print方法,并調(diào)用它。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
package myreflection;
import java.lang.reflect.Method;
public class ReflectionHelloWorld {
public static void main(String[] args){
Foo f = new Foo();
Method method;
try {
method = f.getClass().getMethod("print", new Class<?>[0]);
method.invoke(f);
} catch (Exception e) {
e.printStackTrace();
}
}
}
class Foo {
public void print() {
System.out.println("abc");
}
}
|
輸出:
示例3:創(chuàng)建對(duì)象
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
package myreflection;
public class ReflectionHelloWorld {
public static void main(String[] args){
Class<?> c = null;
try{
c=Class.forName("myreflection.Foo");
}catch(Exception e){
e.printStackTrace();
}
Foo f = null;
try {
f = (Foo) c.newInstance();
} catch (Exception e) {
e.printStackTrace();
}
f.print();
}
}
class Foo {
public void print() {
System.out.println("abc");
}
}
|
示例4:獲取構(gòu)造函數(shù),并創(chuàng)建對(duì)象。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
|
package myreflection;
import java.lang.reflect.Constructor;
public class ReflectionHelloWorld {
public static void main(String[] args){
Class<?> c = null;
try{
c=Class.forName("myreflection.Foo");
}catch(Exception e){
e.printStackTrace();
}
Foo f1 = null;
Foo f2 = null;
Constructor<?> cons[] = c.getConstructors();
try {
f1 = (Foo) cons[0].newInstance();
f2 = (Foo) cons[1].newInstance("abc");
} catch (Exception e) {
e.printStackTrace();
}
f1.print();
f2.print();
}
}
class Foo {
String s;
public Foo(){}
public Foo(String s){
this.s=s;
}
public void print() {
System.out.println(s);
}
}
|
輸出:
此外,你可以通過(guò)Class實(shí)例來(lái)獲取該類實(shí)現(xiàn)的接口、父類、聲明的屬性等。
示例5:通過(guò)反射來(lái)修改數(shù)組的大小。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
|
package myreflection;
import java.lang.reflect.Array;
public class ReflectionHelloWorld {
public static void main(String[] args) {
int[] intArray = { 1, 2, 3, 4, 5 };
int[] newIntArray = (int[]) changeArraySize(intArray, 10);
print(newIntArray);
String[] atr = { "a", "b", "c", "d", "e" };
String[] str1 = (String[]) changeArraySize(atr, 10);
print(str1);
}
public static Object changeArraySize(Object obj, int len) {
Class<?> arr = obj.getClass().getComponentType();
Object newArray = Array.newInstance(arr, len);
int co = Array.getLength(obj);
System.arraycopy(obj, 0, newArray, 0, co);
return newArray;
}
public static void print(Object obj) {
Class<?> c = obj.getClass();
if (!c.isArray()) {
return;
}
System.out.println("
Array length: " + Array.getLength(obj));
for (int i = 0; i < Array.getLength(obj); i++) {
System.out.print(Array.get(obj, i) + " ");
}
}
}
|
輸出:
1
2
3
4
|
Array length: 10
1 2 3 4 5 0 0 0 0 0
Array length: 10
a b c d e null null null null null
|
總結(jié)
上述示例代碼僅僅展現(xiàn)了Java反射機(jī)制很小一部分的功能。如果你覺(jué)得意猶未盡,可以前去閱讀官方文檔。
參考資料:
http://en.wikipedia.org/wiki/Reflection_(computer_programming)
http://docs.oracle.com/javase/tutorial/reflect/
原文鏈接: Programcreek 翻譯: ImportNew.com - 薄荷腦
譯文鏈接: http://www.importnew.com/9078.html
總結(jié)
以上是生活随笔為你收集整理的什么是反射?反射有什么用处的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
如果覺(jué)得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。