本文實(shí)例分析了JAVA反射機(jī)制。分享給大家供大家參考,具體如下:
反射,當(dāng)時(shí)經(jīng)常聽(tīng)他們說(shuō),自己也看過(guò)一些資料,也可能在設(shè)計(jì)模式中使用過(guò),但是感覺(jué)對(duì)它沒(méi)有一個(gè)較深入的了解,這次重新學(xué)習(xí)了一下,感覺(jué)還行吧!
一、先看一下反射的概念:
主要是指程序可以訪問(wèn),檢測(cè)和修改它本身狀態(tài)或行為的一種能力,并能根據(jù)自身行為的狀態(tài)和結(jié)果,調(diào)整或修改應(yīng)用所描述行為的狀態(tài)和相關(guān)的語(yǔ)義。
反射是java中一種強(qiáng)大的工具,能夠使我們很方便的創(chuàng)建靈活的代碼,這些代碼可以再運(yùn)行時(shí)裝配,無(wú)需在組件之間進(jìn)行源代碼鏈接。但是反射使用不當(dāng)會(huì)成本很高!
看概念很暈的,繼續(xù)往下看。
二、反射機(jī)制的作用:
1、反編譯:.class-->.java
2、通過(guò)反射機(jī)制訪問(wèn)java對(duì)象的屬性,方法,構(gòu)造方法等;
這樣好像更容易理解一些,下邊我們具體看怎么實(shí)現(xiàn)這些功能。
三、在這里先看一下sun為我們提供了那些反射機(jī)制中的類(lèi):
1
2
3
4
|
java.lang.Class; java.lang.reflect.Constructor; java.lang.reflect.Field; java.lang.reflect.Method; java.lang.reflect.Modifier; |
很多反射中的方法,屬性等操作我們可以從這四個(gè)類(lèi)中查詢(xún)。還是哪句話(huà)要學(xué)著不斷的查詢(xún)API,那才是我們最好的老師。
四、具體功能實(shí)現(xiàn):
1、反射機(jī)制獲取類(lèi)有三種方法,我們來(lái)獲取Employee類(lèi)型
1
2
3
4
5
6
7
8
9
|
//第一種方式: Classc1 = Class.forName( "Employee" ); //第二種方式: //java中每個(gè)類(lèi)型都有class 屬性. Classc2 = Employee. class ; //第三種方式: //java語(yǔ)言中任何一個(gè)java對(duì)象都有g(shù)etClass 方法 Employeee = new Employee(); Classc3 = e.getClass(); //c3是運(yùn)行時(shí)類(lèi) (e的運(yùn)行時(shí)類(lèi)是Employee) |
2、創(chuàng)建對(duì)象:獲取類(lèi)以后我們來(lái)創(chuàng)建它的對(duì)象,利用newInstance:
1
2
3
|
Class c =Class.forName( "Employee" ); //創(chuàng)建此Class 對(duì)象所表示的類(lèi)的一個(gè)新實(shí)例 Objecto = c.newInstance(); //調(diào)用了Employee的無(wú)參數(shù)構(gòu)造方法. |
3、獲取屬性:分為所有的屬性和指定的屬性:
①、先看獲取所有的屬性的寫(xiě)法:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
//獲取整個(gè)類(lèi) Class c = Class.forName( "java.lang.Integer" ); //獲取所有的屬性? Field[] fs = c.getDeclaredFields(); //定義可變長(zhǎng)的字符串,用來(lái)存儲(chǔ)屬性 StringBuffer sb = new StringBuffer(); //通過(guò)追加的方法,將每個(gè)屬性拼接到此字符串中 //最外邊的public定義 sb.append(Modifier.toString(c.getModifiers()) + " class " + c.getSimpleName() + "{\n" ); //里邊的每一個(gè)屬性 for (Field field:fs){ sb.append( "\t" ); //空格 sb.append(Modifier.toString(field.getModifiers())+ " " ); //獲得屬性的修飾符,例如public,static等等 sb.append(field.getType().getSimpleName() + " " ); //屬性的類(lèi)型的名字 sb.append(field.getName()+ ";\n" ); //屬性的名字+回車(chē) } sb.append( "}" ); System.out.println(sb); |
②、獲取特定的屬性,對(duì)比著傳統(tǒng)的方法來(lái)學(xué)習(xí):
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
public static void main(String[] args) throws Exception{ <span style= "white-space:pre" > </span> //以前的方式: /* User u = new User(); u.age = 12; //set System.out.println(u.age); //get */ //獲取類(lèi) Class c = Class.forName( "User" ); //獲取id屬性 Field idF = c.getDeclaredField( "id" ); //實(shí)例化這個(gè)類(lèi)賦給o Object o = c.newInstance(); //打破封裝 idF.setAccessible( true ); //使用反射機(jī)制可以打破封裝性,導(dǎo)致了java對(duì)象的屬性不安全。 //給o對(duì)象的id屬性賦值"110" idF.set(o, "110" ); //set //get System.out.println(idF.get(o)); } |
4、獲取方法,和構(gòu)造方法,不再詳細(xì)描述,只來(lái)看一下關(guān)鍵字:
方法關(guān)鍵字
含義
getDeclaredMethods()
獲取所有的方法
getReturnType()
獲得方法的放回類(lèi)型
getParameterTypes()
獲得方法的傳入?yún)?shù)類(lèi)型
getDeclaredMethod("方法名",參數(shù)類(lèi)型.class,……)
獲得特定的方法
構(gòu)造方法關(guān)鍵字
含義
getDeclaredConstructors()
獲取所有的構(gòu)造方法
getDeclaredConstructor(參數(shù)類(lèi)型.class,……)
獲取特定的構(gòu)造方法
父類(lèi)和父接口
含義
getSuperclass()
獲取某類(lèi)的父類(lèi)
getInterfaces()
獲取某類(lèi)實(shí)現(xiàn)的接口
這樣我們就可以獲得類(lèi)的各種內(nèi)容,進(jìn)行了反編譯。對(duì)于JAVA這種先編譯再運(yùn)行的語(yǔ)言來(lái)說(shuō),反射機(jī)制可以使代碼更加靈活,更加容易實(shí)現(xiàn)面向?qū)ο蟆?/p>
五、反射加配置文件,使我們的程序更加靈活:
在設(shè)計(jì)模式學(xué)習(xí)當(dāng)中,學(xué)習(xí)抽象工廠的時(shí)候就用到了反射來(lái)更加方便的讀取數(shù)據(jù)庫(kù)鏈接字符串等,當(dāng)時(shí)不是太理解,就照著抄了。看一下.NET中的反射+配置文件的使用:
當(dāng)時(shí)用的配置文件是app.config文件,內(nèi)容是XML格式的,里邊填寫(xiě)鏈接數(shù)據(jù)庫(kù)的內(nèi)容:
1
2
3
4
5
|
< configuration > < appSettings > < add key = "" value = "" /> </ appSettings > </ configuration > |
反射的寫(xiě)法:
assembly.load("當(dāng)前程序集的名稱(chēng)").CreateInstance("當(dāng)前命名空間名稱(chēng)".要實(shí)例化的類(lèi)名);
這樣的好處是很容易的方便我們變換數(shù)據(jù)庫(kù),例如我們將系統(tǒng)的數(shù)據(jù)庫(kù)從SQL Server升級(jí)到Oracle,那么我們寫(xiě)兩份D層,在配置文件的內(nèi)容改一下,或者加條件選擇一下即可,帶來(lái)了很大的方便。
當(dāng)然了,JAVA中其實(shí)也是一樣,只不過(guò)這里的配置文件為.properties,稱(chēng)作屬性文件。通過(guò)反射讀取里邊的內(nèi)容。這樣代碼是固定的,但是配置文件的內(nèi)容我們可以改,這樣使我們的代碼靈活了很多!
綜上為,JAVA反射的再次學(xué)習(xí),靈活的運(yùn)用它,能夠使我們的代碼更加靈活,但是它也有它的缺點(diǎn),就是運(yùn)用它會(huì)使我們的軟件的性能降低,復(fù)雜度增加,所以還要我們慎重的使用它。
希望本文所述對(duì)大家Java程序設(shè)計(jì)有所幫助。