欧美成人高清,97精品国产福利一区二区三区,аⅴ资源天堂资源库在线,色999日韩自偷自拍美女

到底什么是反射?反射究竟能用來干什么?(到底什么是反射-反射究竟能用來干什么呢)

到底什么是反射?反射究竟能用來干什么?(到底什么是反射-反射究竟能用來干什么呢)

反射,顧名思義,它是一種逆向的操作。就好像人在照鏡子的時候,正是由于光的反射,才能看到鏡子中的自己。而在Java中,反射功能就好比是一面鏡子,通過它,我們可以在程序運行過程中看到Class以及對象的相關(guān)信息。

在以往的經(jīng)驗中,當我們需要完成某些操作,往往是在編譯之前完成,比如根據(jù)創(chuàng)建對象、讀取屬性、設置屬性;我們把這些程序編寫完之后編譯器會將之編譯為class文件,然后直接在虛擬機中運行就可以了,大部分情況確實是這樣,這也是為什么Java是"靜態(tài)語言"。

但是我們卻可以通過反射來完成在動態(tài)語言中才能做到的一些操作,比如首先第一步,通過反射獲取某個.class文件的結(jié)構(gòu)信息。

示例如下:

/** * 蝙蝠俠 */public class Batman { public Batman(String name,int age,String power){ this.name = name; this.age = age; this.power = power; } //蝙蝠俠的名字 private String name; //蝙蝠俠的年齡 private int age; //蝙蝠俠的超能力 private String power; public String getName() { return name; } public void setName(String name) { this.name = name; } public int getAge() { return age; } public void setAge(int age) { this.age = age; } public String getPower() { return power; } public void setPower(String power) { this.power = power; } /** 跟小貓談戀愛 */ private void beInLove(){ System.out.println("蝙蝠俠正在跟小貓談戀愛"); } /** 教訓小丑 */ public void work(){ System.out.println("蝙蝠俠正在教訓小丑"); } private void workWithGordon(String name){ System.out.printf("蝙蝠俠正在和%s一起教訓小丑",name); }}復制代碼

//測試類public class ReflectTest { public static void main(String[] args) { Class cls = Batman.class; //1.獲取class中的所有屬性,包括全局屬性和局部屬性 Field[] fields = cls.getDeclaredFields(); System.out.println("屬性:"); for (Field f : fields) { System.out.println(f); } System.out.println("方法:"); //獲取class中的所有方法 Method[] mets = cls.getDeclaredMethods(); for (Method met : mets) { System.out.println(met); } System.out.println("構(gòu)造器:"); //獲取class中的所有構(gòu)造器 Constructor[] cons = cls.getDeclaredConstructors(); for (Constructor c : cons) { System.out.println(c); } }}復制代碼

運行結(jié)果:
屬性:
private java.lang.String day2.demo2.Batman.name private int day2.demo2.Batman.age private java.lang.String day2.demo2.Batman.power
方法:
public int day2.demo2.Batman.getAge() public void day2.demo2.Batman.setAge(int) public java.lang.String day2.demo2.Batman.getPower() public void day2.demo2.Batman.setPower(java.lang.String) public java.lang.String day2.demo2.Batman.getName() public void day2.demo2.Batman.setName(java.lang.String)
構(gòu)造器:
public day2.demo2.Batman(java.lang.String,int,java.lang.String)

java程序在運行時,虛擬機在加載類時,會為這個類創(chuàng)建一個Class對象,用來表示這個類的信息。可以通過類名.class、Class.forName("類名")、Object.getClass等方式獲取到一個Class對象,這個對象記錄了類的信息,通過它可以逆向獲取類的結(jié)構(gòu)。

在示例中,通過Class對象中方法的調(diào)用,獲取了Batman類中的所有屬性、方法、構(gòu)造器,但是反射的功能遠不止于此,比如通過上述三個方法獲取到的Field、Method、Constructor數(shù)組對象完成進一步的操作:

public class ReflectTest { public static void main(String[] args) { Class cls = Batman.class; Field[] fields = cls.getDeclaredFields(); System.out.println("屬性:"); System.out.println("訪問修飾符:" Modifier.toString(fields[0].getModifiers())); System.out.println("是否靜態(tài)的:" Modifier.isStatic(fields[0].getModifiers())); System.out.println("是否為public:" Modifier.isPublic(fields[0].getModifiers())); System.out.println("是否常量:" Modifier.isFinal(fields[0].getModifiers())); System.out.println("方法:"); Method[] mets = cls.getDeclaredMethods(); System.out.println("是否為本地方法:" Modifier.isNative(mets[0].getModifiers())); System.out.println("是否為抽象方法:" Modifier.isAbstract(mets[0].getModifiers())); System.out.println("是否為接口:" Modifier.isInterface(mets[0].getModifiers())); System.out.println("是否線程同步:" Modifier.isSynchronized(mets[0].getModifiers())); System.out.println("構(gòu)造器:"); Constructor[] cons = cls.getDeclaredConstructors(); System.out.println("是否公有:" Modifier.isPublic(cons[0].getModifiers())); /** * ... * */ }}復制代碼

以上這些都是直接對Class類的操作,其實java反射也同樣支持對運行中的對象的操作,甚至可以修改對象中屬性的值。

示例代碼:

public class ReflectTest { public static void main(String[] args) throws IllegalAccessException { Batman batman = new Batman("布魯斯韋恩",27,"有錢"); Class cls = batman.getClass(); Field[] fields = cls.getDeclaredFields(); /**獲取第一個屬性name的值,由于是private屬性, 所以會報IllegalAccessException異常,很顯然是與權(quán)限有關(guān)*/ try { var name = fields[0].get(batman); System.out.println(name); }catch (IllegalAccessException e){ //這里通過一個方法設置可訪問對象的可訪問標志 fields[0].setAccessible(true); var name = fields[0].get(batman); System.out.println(name); } //但是這里設置的只是數(shù)組中第一個屬性的訪問權(quán)限,下面這句話依然會報錯 try { var name = fields[1].get(batman); System.out.println(name); }catch (IllegalAccessException e){ //所以通過下面的方法對整個數(shù)組對象的訪問權(quán)限進行設置 AccessibleObject.setAccessible(fields,true); var age = fields[1].get(batman); var power = fields[2].get(batman); System.out.println(age); System.out.println(power); } //修改fields[2]的值 fields[2].set(batman,"哥譚首富"); System.out.println("超能力:" batman.getPower()); }}復制代碼

運行結(jié)果:
布魯斯韋恩
27
有錢
超能力:哥譚首富

通過調(diào)用對象的getClass()方法獲取這個類唯一的Class對象,再通過獲取到field對象的get(obj)方法獲取到這個field的值(當然如果屬性是私有的,還需要使用setAccessible方法設置訪問標志),并且不僅可以獲取,還能通過其set(obj,val)方法重新設置這個屬性的值。而這一切都是在程序運行期間完成的,成功的通過反射修改了對象中的屬性。

至此,已經(jīng)實現(xiàn)了通過反射來查看類的信息、對象的屬性以及設置對象的屬性。那么如何通過反射來調(diào)用方法以及構(gòu)造器呢?

示例代碼:

public class ReflectTest { public static void main(String[] args) throws IllegalAccessException, NoSuchMethodException, InvocationTargetException, NoSuchFieldException, InstantiationException { Class cls = Batman.class; //通過反射調(diào)用構(gòu)造器創(chuàng)建蝙蝠俠對象 Batman batman = (Batman) cls.getDeclaredConstructor(String.class,int.class,String.class).newInstance("蝙蝠俠", 27, "有錢"); Method method = cls.getDeclaredMethod("beInLove"); //因為beInLove()方法是私有的,所以需要設置以下權(quán)限 method.setAccessible(true); method.invoke(batman); Method work = cls.getDeclaredMethod("work"); //work方法不是private的,不需要設置權(quán)限 work.invoke(batman); //調(diào)用帶參數(shù)的方法 Method workWithGordon = cls.getDeclaredMethod("workWithGordon", String.class); //私有方法依然要設置權(quán)限 workWithGordon.setAccessible(true); workWithGordon.invoke(batman,"Gordon"); }}復制代碼

至此,就完成了方法及構(gòu)造方法的調(diào)用。需要注意的是,若調(diào)用了一個帶返回值的方法,如果返回值類型是基本類型,invoke方法會返回其包裝類型,如int返回Integer、double返回Double。

另外,java.lang.reflect包中還提供了一個很好用的類Array,ArrayList中的數(shù)組擴容就使用到了這個類。

示例代碼:

public class ReflectTest { public static void main(String[] args) throws IllegalAccessException, NoSuchMethodException, InvocationTargetException, NoSuchFieldException, InstantiationException { //假如我想創(chuàng)建一個數(shù)組 String[] strs = {"bruce","jack","jerry"}; strs = (String[]) copyOf(strs,10); } //現(xiàn)在我想寫一個方法來為泛型數(shù)組擴容 public static Object[] CopyOf(Object[] obj,int nlength){ var newArray = new Object[nlength]; System.arraycopy(obj,0,newArray,0,Math.min(obj.length,nlength)); return newArray; }}復制代碼

運行結(jié)果:Exception in thread "main" java.lang.ClassCastException

這段代碼看起來好像沒有問題,通過Object超類接收任意類型的數(shù)組。但是有一個細節(jié)問題,當創(chuàng)建一個數(shù)組然后將其轉(zhuǎn)為Object[],再把它從Object[]轉(zhuǎn)回來是沒有問題的,但是如果直接創(chuàng)建一個Object[]轉(zhuǎn)成目標類型的數(shù)組是會出錯的。所以上述代碼無法完成泛型數(shù)組的擴容。

現(xiàn)在對代碼做一些改進,示例代碼:

public class ReflectTest { public static void main(String[] args) throws IllegalAccessException, NoSuchMethodException, InvocationTargetException, NoSuchFieldException, InstantiationException { //假如我想創(chuàng)建一個數(shù)組 String[] strs = {"bruce","jack","jerry"}; strs = (String[]) CopyOf(strs,10); System.out.println(strs.length); } //現(xiàn)在我想寫一個方法來為泛型數(shù)組擴容 public static Object CopyOf(Object obj,int nlength){ Class cls = obj.getClass(); if(!cls.isArray()) return null; //獲取數(shù)組的類型 Class type = cls.getComponentType(); //獲取數(shù)組的長度 int length = Array.getLength(obj); //通過Array.newInstance創(chuàng)建一個泛型數(shù)組,類型通過參數(shù)指定 Object newArray = Array.newInstance(type,nlength); System.arraycopy(obj,0,newArray,0,Math.min(length,nlength)); return newArray; }}復制代碼

運行結(jié)果:10

這次程序成功運行,并且成功為數(shù)組擴容。最主要的原因是代碼中的關(guān)鍵方法,Array類的靜態(tài)方法newInstance,這個方法能夠返回一個有給定類型,給定大小的新數(shù)組,而不是一個簡單的Object[]。

總結(jié):反射機制可以在運行時查看、操作字段和方法。但是不應該濫用反射,因為反射在編譯階段無法查找出錯誤,如果存在問題,往往到了運行時才會發(fā)現(xiàn)。JVM無法對反射的相關(guān)代碼做優(yōu)化,所以效率相對低。并且反射可能導致程序不安全。

作者:現(xiàn)在沒有牛仔了
鏈接:https://juejin.cn/post/7228967103349080120

版權(quán)聲明:本文內(nèi)容由互聯(lián)網(wǎng)用戶自發(fā)貢獻,該文觀點僅代表作者本人。本站僅提供信息存儲空間服務,不擁有所有權(quán),不承擔相關(guān)法律責任。如發(fā)現(xiàn)本站有涉嫌抄襲侵權(quán)/違法違規(guī)的內(nèi)容, 請發(fā)送郵件至89291810@qq.com舉報,一經(jīng)查實,本站將立刻刪除。
(0)
上一篇 2024年4月24日 下午2:05
下一篇 2024年4月24日 下午2:11

相關(guān)推薦

  • 私立學校不去學校還用交學費嗎

    私立學校不去學校還用交學費嗎? 近年來,私立學校已經(jīng)成為許多家長選擇學校的首選。私立學校通常會提供更好的教育資源和更高質(zhì)量的教學,因此許多家長都希望孩子能夠在私立學校中成長。但是,…

    教育百科 2024年11月13日
  • 中北大學2025年休學政策

    中北大學2025年休學政策 隨著時代的不斷發(fā)展和進步,人們的生活方式和工作方式也在不斷改變。為了更好地適應這種變化,許多學生選擇休學一段時間,調(diào)整自己的心態(tài)和狀態(tài),以適應新的環(huán)境和…

    教育百科 2024年11月3日
  • 上海初中休學教育局審核多長時間批下來

    上海初中休學教育局審核多長時間批下來 休學是指學生因疾病、家庭緊急情況等原因需要暫停學習,以便治療或恢復健康。在上海,初中學生休學的情況較為普遍。然而,休學申請的審核時間也是一個令…

    教育百科 2024年10月29日
  • 青島初中學校2025年最新排名

    青島初中學校2025年最新排名 隨著教育的不斷發(fā)展和進步,青島的初中學校排名也在不斷變化。根據(jù)最新的排名,以下是青島初中學校2025年的最新排名: 1. 青島市第五中學2. 青島中…

    教育百科 2025年1月21日
  • 鄭州港區(qū)最好的高中是哪個

    鄭州港區(qū)最好的高中是哪個? 在鄭州港區(qū),有許多優(yōu)秀的高中,但是最好的高中是哪個,這個問題并沒有一個確切的答案。不過,我們可以從以下幾個方面來考慮這個問題: 1. 師資力量 師資力量…

    教育百科 2024年11月19日
  • 3歲孩子可以玩的游戲孩子光玩手機

    孩子光玩手機是一個讓人擔憂的問題。現(xiàn)在,許多三歲的孩子沉迷于手機和其他電子設備,而不是參加戶外運動或與同齡人玩耍。這種行為可能會導致孩子缺乏社交技能,注意力缺陷多動障礙(ADHD)…

    教育百科 2024年8月19日
  • 怎樣幫孩子戒網(wǎng)癮如何拜托厭學

    幫助孩子戒網(wǎng)癮和拜托厭學,是每個父母都應該盡力去做的一件事情。網(wǎng)癮和厭學可能會對孩子的學習成績,社交能力,心理健康造成嚴重影響。為了幫助孩子們克服這些問題,我們可以采取以下措施: …

    教育百科 2024年10月8日
  • 朋友不上學

    我的朋友不上學 我的朋友小明是一名初中生,他最近總是表現(xiàn)出一些奇怪的行為。他不再像以前一樣按時上學,也不再像以前一樣認真學習。他時常獨自一人在房間里,看著電視或電腦,好像忘記了自己…

    教育百科 2025年4月7日
  • 為什么重慶市簡稱“渝”、“巴”呢?(重慶市為什么簡稱為渝)

    重慶市,位于中國西南部,是四大直轄市之一。重慶是國家歷史文化名城,有3000余年歷史,亦是巴渝文化的發(fā)祥地;因嘉陵江古稱“渝水”,故簡稱“渝”;又因宋淳熙十六年,光宗趙惇先封恭王再…

    教育百科 2024年4月3日
  • 心中有丘壑眉目作山河下一句

    心中有丘壑,眉目作山河。這句話出自唐代詩人王之渙的《登鸛雀樓》。它描繪了一個富有詩意的場景,即人心中充滿了對于世界的理解和感悟,這些感悟如同山川河流一般,在人的眉目間展現(xiàn)出一種獨特…

    教育百科 2025年2月5日

發(fā)表回復

您的郵箱地址不會被公開。 必填項已用 * 標注

主站蜘蛛池模板: 辽宁省| 张家口市| 玉林市| 甘洛县| 阿瓦提县| 翼城县| 读书| 香河县| 阜新市| 大名县| 蒲城县| 河南省| 大宁县| 浏阳市| 新干县| 基隆市| 革吉县| 淅川县| 万载县| 沾益县| 九江县| 礼泉县| 静海县| 兰考县| 丹寨县| 台中县| 鄄城县| 临沭县| 贵定县| 麻江县| 司法| 中西区| 阿巴嘎旗| 江西省| 五河县| 繁峙县| 德令哈市| 稻城县| 沂源县| 临汾市| 牟定县|