Java反射機(jī)制-較全面_第1頁
Java反射機(jī)制-較全面_第2頁
Java反射機(jī)制-較全面_第3頁
Java反射機(jī)制-較全面_第4頁
Java反射機(jī)制-較全面_第5頁
已閱讀5頁,還剩19頁未讀 繼續(xù)免費(fèi)閱讀

下載本文檔

版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請(qǐng)進(jìn)行舉報(bào)或認(rèn)領(lǐng)

文檔簡介

Java反射機(jī)制2021/6/27st,2009概述本課程主要講述Java反射機(jī)制本課程要求大家對(duì)Java泛型知識(shí)有所了解,因?yàn)槌绦虼a中大量使用了泛型相關(guān)知識(shí)2021/6/272目錄Java反射簡介………4ClassObject

………8動(dòng)態(tài)實(shí)例化………11Method使用………14Field使用………16實(shí)用案例………18總結(jié)………222021/6/273動(dòng)態(tài)語言“程序運(yùn)行時(shí),允許改變程序結(jié)構(gòu)或變量類型,這種語言稱為動(dòng)態(tài)語言”。從這個(gè)觀點(diǎn)看,Perl,Python,Ruby是動(dòng)態(tài)語言,C++,Java,C#不是動(dòng)態(tài)語言。盡管在這樣的定義與分類下Java不是動(dòng)態(tài)語言,它卻有著一個(gè)非常突出的動(dòng)態(tài)相關(guān)機(jī)制:Reflection。2021/6/274什么是反射反射的概念是由Smith在1982年首次提出的,主要是指程序可以訪問、檢測和修改它本身狀態(tài)或行為的一種能力。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語言的反射機(jī)制。2021/6/275Java反射的應(yīng)用Spring框架:IOC(控制反轉(zhuǎn))Hibernate框架:關(guān)聯(lián)映射等白盒測試2021/6/276Java反射相關(guān)的APIjava.lang包下Class<T>:表示一個(gè)正在運(yùn)行的Java應(yīng)用程序中的類和接口,是Reflection的起源java.lang.reflect包下Field類:代表類的成員變量(也稱類的屬性)Method類:代表類的方法Constructor類:代表類的構(gòu)造方法Array類:提供了動(dòng)態(tài)創(chuàng)建數(shù)組,以及訪問數(shù)組的元素的靜態(tài)方法2021/6/277Class<T>類是程序的一部分,每個(gè)類都有一個(gè)Class對(duì)象。換言之,每當(dāng)編寫并且編譯了一個(gè)新類,就會(huì)產(chǎn)生一個(gè)Class對(duì)象Class沒有公共構(gòu)造方法。Class對(duì)象是在加載類時(shí)由Java虛擬機(jī)以及通過調(diào)用類加載器中的defineClass方法自動(dòng)構(gòu)造的,因此不能顯式地聲明一個(gè)Class對(duì)象Class是Reflection起源。要想操縱類中的屬性和方法,都必須從獲取Classobject開始2021/6/278第一個(gè)實(shí)例就用大家非常熟悉的ArrayList類,我們嘗試來獲取ArrayList申明的方法。 publicstaticvoidmain(String[]args){ ArrayListaList=newArrayList(); ClassalClass=aList.getClass(); System.out.println("①"+alClass); System.out.println("②"+alClass.getName()); Method[]alMethod=alClass.getDeclaredMethods(); for(Methodmethod:alMethod){ System.out.println("③"+method); System.out.println("④"+method.getName()); } }第一步永遠(yuǎn)是獲得被反射類的Class對(duì)象!案例一2021/6/279獲取ClassObject獲取方式說明示例object.getClass()每個(gè)對(duì)象都有此方法獲取指定實(shí)例對(duì)象的ClassListlist=newArrayList();ClasslistClass=list.getClass();class.getSuperclass()獲取當(dāng)前Class的繼承類ClassListlist=newArrayList();ClasslistClass=list.getClass();ClasssuperClass=listClass.getSuperclass();Object.class.class直接獲取ClasslistClass=ArrayList.class;Class.forName(類名)用Class的靜態(tài)方法,傳入類的全稱即可try{Classc=Class.forName("java.util.ArrayList");}catch(ClassNotFoundExceptione){e.printStackTrace();}Primitive.TYPE基本數(shù)據(jù)類型的封裝類獲取Class的方式ClasslongClass=Long.TYPE;ClassintegerClass=Integer.TYPE;ClassvoidClass=Void.TYPE;根據(jù)具體情形和個(gè)人愛好,可以選擇下面任何一種方式獲得Class對(duì)象2021/6/2710通過反射實(shí)例化對(duì)象平常情況我們通過newObject來生成一個(gè)類的實(shí)例,但有時(shí)候我們沒法直接new,只能通過反射動(dòng)態(tài)生成。實(shí)例化無參構(gòu)造函數(shù)的對(duì)象,兩種方式:①Class.newInstance();②Class.getConstructor(newClass[]{}).newInstance(newObject[]{})實(shí)例化帶參構(gòu)造函數(shù)的對(duì)象:clazz.getConstructor(Class<?>...

parameterTypes).newInstance(Object...

initargs)2021/6/2711案例準(zhǔn)備首先我們新建一個(gè)JavaBean—User,User繼承自另一個(gè)Bean—BaseUser。注意:這兩個(gè)Bean的屬性和方法的作用域!2021/6/2712案例二:動(dòng)態(tài)實(shí)例化2021/6/2713通過反射調(diào)用Method(方法)獲得當(dāng)前類以及超類的publicMethod:Method[]arrMethods=classType.getMethods();獲得當(dāng)前類申明的所有Method:Method[]arrMethods=classType.getDeclaredMethods();獲得當(dāng)前類以及超類指定的publicMethod:Methodmethod=classType.getMethod(String

name,Class<?>...

parameterTypes);獲得當(dāng)前類申明的指定的Method:Methodmethod=classType.getDeclaredMethod(Stringname,Class<?>...

parameterTypes)通過反射動(dòng)態(tài)運(yùn)行指定Method:Objectobj=method.invoke(Object

obj,Object...

args)2021/6/2714案例三:動(dòng)態(tài)操縱Method2021/6/2715通過反射調(diào)用Field(變量)獲得當(dāng)前類以及超類的publicField:Field[]arrFields=classType.getFields();獲得當(dāng)前類申明的所有Field:Field[]arrFields=classType.getDeclaredFields();獲得當(dāng)前類以及超類指定的publicField:Fieldfield=classType.getField(String

name);獲得當(dāng)前類申明的指定的Field:Fieldfield=classType.getDeclaredField(Stringname);通過反射動(dòng)態(tài)設(shè)定Field的值:fieldType.set(Objectobj,Object

value);通過反射動(dòng)態(tài)獲取Field的值:

Objectobj=fieldType.get(Object

obj);2021/6/2716案例四:動(dòng)態(tài)操縱Field2021/6/2717案例五:趁熱打鐵(提出問題)在Hibernate中,已知有一個(gè)user實(shí)體(屬性id,name,phone)需要被update,我們通常有三種方式:①首先UserloadUser=session.load(user.getId);此時(shí)loadUser是持久化的,然后使用loadUser.setX(user.getX)方法把需要更新的字段set一下②寫hql語句③session.update(user);

問題來了:假如user實(shí)體中只有id和name有值,如果我們用以上方式更新的話,phone因?yàn)槭莕ull,數(shù)據(jù)庫的phone本來是有值的,但經(jīng)過更新后,也會(huì)被更新成null。

那么有什么方法能判斷user實(shí)體中哪些對(duì)象為null呢?然后我們就可以不更新那些字段。也許反射可以幫忙解決。2021/6/2718案例五:趁熱打鐵(分析問題)已知有一個(gè)user實(shí)體(屬性id,name,phone)需要被update我們的解決方式其實(shí)很簡單:首先UserloadUser=session.load(user.getId);此時(shí)loadUser是持久化的然后使用loadUser.setXXX(user.getXXX)方法把需要更新的字段set一下至于怎么判斷哪些屬性需要更新,我們可以通過反射先獲得所有的getXXX方法,然后逐個(gè)invoke獲得它們的值,判斷一下如果值需要更新才執(zhí)行l(wèi)oadUser.setXX(user.getXXX)2021/6/2719案例五:趁熱打鐵(解決問題)看源碼:2021/6/2720Spring框架的IOC的簡化實(shí)現(xiàn)2021/6/2721Java反射總結(jié)只要用到反射,先獲得ClassObject沒有方法能獲得當(dāng)前類的超類的private方法和屬性,你必須通過getSuperclass()找到超類以后再去嘗試獲得通常情況即使是當(dāng)前類,privat

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請(qǐng)下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請(qǐng)聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 人人文庫網(wǎng)僅提供信息存儲(chǔ)空間,僅對(duì)用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對(duì)用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對(duì)任何下載內(nèi)容負(fù)責(zé)。
  • 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請(qǐng)與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶因使用這些下載資源對(duì)自己和他人造成任何形式的傷害或損失。

評(píng)論

0/150

提交評(píng)論