成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Java 反射:讓你更優(yōu)雅的使用框架!

開發(fā)
毫不夸張地說,沒有反射,很多優(yōu)秀的框架不復存在,沒有這些優(yōu)秀的框架(比如Spring),Java可能會遜色很多,因此,這篇文章,我們一起來深入探討Java反射以及其背后的原理。

在 Java語言中,反射是一種強大而優(yōu)秀的機制,通過反射,我們可以在運行時檢查和修改類、接口、字段和方法的信息,甚至動態(tài)地創(chuàng)建對象、調(diào)用方法和訪問私有成員。

可以毫不夸張地說,沒有反射,很多優(yōu)秀的框架不復存在,沒有這些優(yōu)秀的框架(比如Spring),Java可能會遜色很多,因此,這篇文章,我們一起來深入探討Java反射以及其背后的原理。

一、什么是反射

先看看 Oracle官方對java反射的說明:

Reflection enables Java code to discover information about the fields, methods and constructors of loaded classes, and to use reflected fields, methods, and constructors to operate on their underlying counterparts, within security restrictions. The API accommodates applications that need access to either the public members of a target object (based on its runtime class) or the members declared by a given class. It also allows programs to suppress default reflective access control.

Java 的反射機制是指在運行狀態(tài)中,對于任意一個類都能夠知道這個類所有的屬性和方法;并且對于任意一個對象,都能夠調(diào)用它的任意一個方法;這種動態(tài)獲取信息以及動態(tài)調(diào)用對象方法的功能成為Java語言的反射機制。

它是通過 Java反射 API 來實現(xiàn),其中最核心的類位于 java.lang.reflect 包下,如 Class、Constructor、Field 和 Method等,這些類提供了對類和對象的運行時信息進行檢查和操作的方法。如下圖,展示了 JDK源碼中 java.lang.reflect 包所有的類:

二、反射的原理

反射的原理主要可以從下面 4個點來闡述:

  • 類加載:當 Java程序運行時,類加載器會根據(jù)類的名稱查找并加載類的字節(jié)碼文件,然后將字節(jié)碼文件轉(zhuǎn)換為可執(zhí)行的 Java類,并將其存儲在運行時數(shù)據(jù)區(qū)域的方法區(qū)中。
  • 創(chuàng)建 Class對象:在類加載過程中,Java虛擬機會自動創(chuàng)建對應的Class對象,Class對象包含了類的元數(shù)據(jù)信息,并提供了訪問和操作類的接口。
  • 獲取 Class對象:Class對象通過多種方式獲取,最常見的方式有 3種: 類的 .class屬性、類實例的 getClass()方法、Class.forName()。
  • 訪問和操作:通過Class對象獲取類的字段、方法、構造函數(shù)等信息,使用Field類和Method類來訪問和操作字段和方法,甚至可以調(diào)用私有的字段和方法。

通過上述的分析可以看出:反射機制需要基于Java虛擬機對類的加載、存儲和訪問機制的支持,通過反射,可以在運行時動態(tài)地探索和操作類的信息,實現(xiàn)靈活的編程和代碼的動態(tài)行為。

三、如何使用反射

在講解了 Java反射原理之后,我們通過一個真實的例子來展示如何使用 Java反射機制。如下示例 demo,通過反射給 Person 類中的 greet() 方法傳入一個 name,然后輸出:

過程分析:

  • 首先,在示例代碼通過獲Person.class取了 Person的Class對象;
  • 然后,使用clazz.getName()獲取了類的名稱,通過clazz.getModifiers()獲取了類的修飾符,并打印輸出;
  • 接下來,通過clazz.getDeclaredMethods()獲取類的所有方法,并依次打印輸出方法的名稱;
  • 接著,通過clazz.getDeclaredConstructor().newInstance()方法創(chuàng)建了 Person 的實例;
  • 再接著,使用clazz.getDeclaredMethod()方法獲取了 greet()方法的引用。為了調(diào)用私有方法,我們需要調(diào)用setAccessible(true)來設置方法的可訪問性。
  • 最后,使用Method.invoke()方法調(diào)用了 greet()方法,傳遞參數(shù)name = Java。

運行示例結果如下圖:

上述示例,我們通過詳細的步驟展示了如何使用反射獲取類的信息和動態(tài)調(diào)用方法。你也可以嘗試在 Person 中添加更多的方法和字段,并使用反射來獲取和操作它們。

四、部分源碼解讀

在上述示例講解時,最后是調(diào)用 Method.invoke() 實現(xiàn) Person.greet()的調(diào)用,因此,這里我們主要分析 invoke()方案,官方源碼截圖:

從上面源碼截圖看出:Method.invoke() 方法,真實返回的是接口 MethodAccessor.invoke()方法。MethodAccessor 接口有三個實現(xiàn)類,具體是調(diào)用哪個類的 invoke 方法?

進入acquireMethodAccessor方法,可以看到MethodAccessor由ReflectionFactory 的 newMethodAccessor方法決定。

再進入 DelegatingMethodAccessorImpl 的 invoke方法:

DelegatingMethodAccessorImpl的invoke方法返回的是MethodAccessorImpl的invoke方法,而MethodAccessorImpl的invoke方法,由它的子類NativeMethodAccessorImpl重寫,這時候返回的是native invoke0,如下圖:

跟到源碼最后可以發(fā)現(xiàn):Method.invoke()方法最終調(diào)用 native的invoke0(),應用層面的操作最終轉(zhuǎn)換成對操作系統(tǒng) c/c++方法的調(diào)用。

五、反射優(yōu)缺點

上面內(nèi)容的講解已經(jīng)側(cè)面反映出了Java反射的一些優(yōu)點,這里再詳細的總結下反射的優(yōu)缺點:

優(yōu)點:

  • 動態(tài)性:反射允許我們在運行時動態(tài)地獲取和操作類的信息,而不需要在編譯時確定。這為編寫靈活的、可擴展的代碼提供了便利。
  • 靈活性:通過反射,我們可以繞過訪問修飾符的限制,訪問和修改私有成員、調(diào)用私有方法等。這為我們在特殊情況下進行一些高級操作提供了可能。
  • 框架開發(fā):反射在開發(fā)框架和庫時非常有用。通過反射,框架可以動態(tài)地加載和實例化類,解析注解,處理回調(diào)等。這為框架提供了更大的靈活性和可擴展性。
  • 調(diào)試和探索:反射使得我們可以在運行時探索代碼背后的信息,例如獲取類的結構、方法、字段等。這對于調(diào)試和理解復雜的代碼非常有幫助。

缺點:

  • 性能開銷:相比于直接調(diào)用代碼,使用反射會帶來更高的性能開銷。反射涉及到動態(tài)查找、方法調(diào)用等操作,這些操作比直接調(diào)用代碼更加耗時。因此,在對性能要求較高的場景下,過度使用反射可能導致性能下降。
  • 安全性和穩(wěn)定性:反射打破了封裝性和類型安全性,通過反射,我們可以繞過訪問修飾符的限制,調(diào)用私有方法等,這可能導致代碼的不穩(wěn)定性和安全隱患。因此,使用反射時需要格外小心,確保代碼的正確性和穩(wěn)定性。

從整體上看,Java反射是以犧牲了小部分的性能換取了更好的擴展性和靈活性,犧牲小我成就大我,而且,隨著現(xiàn)代硬件設備能力越來越強,這點小性能的犧牲是完全值得的。

六、為什么需要反射 

反射機制在 Java中的作用不言而喻,下面列舉了反射機制的一些常見場景和原因:

  • 運行時類型檢查:反射機制允許在運行時獲取類的信息,包括字段、方法和構造方法等。因此,在進行運行時類型檢查,以確保代碼在處理不同類型的對象時能夠正確地進行操作。
  • 動態(tài)創(chuàng)建對象:通過反射,可以在運行時動態(tài)地創(chuàng)建對象,而不需要在編譯時知道具體的類名。這對于某些需要根據(jù)條件或配置來創(chuàng)建對象的情況非常有用,例如工廠模式或依賴注入框架。
  • 訪問和修改私有成員:反射機制可以繞過訪問權限限制,訪問和修改類的私有字段和方法。雖然這破壞了封裝性原則,但在某些特定情況下,這種能力可以幫助我們進行一些特殊操作,例如單元測試、調(diào)試或框架的內(nèi)部實現(xiàn)。
  • 動態(tài)調(diào)用方法:反射機制允許我們在運行時動態(tài)地調(diào)用類的方法,甚至可以根據(jù)運行時的條件來選擇不同的方法。這對于實現(xiàn)插件化系統(tǒng)、處理回調(diào)函數(shù)或?qū)崿F(xiàn)動態(tài)代理等功能非常有用。
  • 框架和庫的實現(xiàn):許多Java框架和庫在其實現(xiàn)中廣泛使用了反射機制。它們利用反射來自動發(fā)現(xiàn)和加載類、實現(xiàn)依賴注入、處理注解、配置文件解析和動態(tài)代理等。反射機制使得這些框架和庫更加靈活和擴展。

七、常用框架 

很多優(yōu)秀的框架內(nèi)部都使用了Java反射,這里重點講解下給 Java打下半壁江山的 Spring生態(tài)(Spring Framework,Spring MVC,SpringBoot, SpringCloud...),以 Spring Framework為例:

  • 依賴注入(Dependency Injection) : 依賴注入,可以把程序員主動創(chuàng)建對象的事情交給 Spring管理,大大提升了對象創(chuàng)建的靈活性。當我們在配置文件或用注解定義 Bean時,Spring會使用反射來動態(tài)地實例化對象,并將依賴的其他對象注入到這些實例中。
  • 自動裝配(Autowired) : 當 Spring容器啟動時,它會掃描應用程序中的所有類,并使用反射來查找和識別帶有 @Autowired注解的字段、方法或構造函數(shù)。再自動將 Bean注入到需要的位置,實現(xiàn)對象之間的自動連接。
  • AOP(Aspect-Oriented Programming) : AOP 利用了動態(tài)代理和反射機制。通過定義切面(Aspect)和切點(Pointcut),Spring可以在運行時使用反射來創(chuàng)建代理對象,從而實現(xiàn)橫切關注點(cross-cutting concerns)的功能,如日志記錄、事務管理等。
  • 動態(tài)代理(Dynamic Proxy) : Spring利用 Java反射機制動態(tài)地創(chuàng)建代理對象,并在代理對象中添加額外的邏輯,從而實現(xiàn)對目標對象的增強。
  • 框架擴展和定制: Spring通過反射機制來實現(xiàn)對應用程序的擴展和定制的。例如,Spring提供了BeanPostProcessor接口,允許開發(fā)人員在 Bean初始化前后插入自定義邏輯,這是通過反射來實現(xiàn)的。

另外,還有一些耳熟能詳?shù)目蚣芤彩褂昧薐ava反射:

  • JUnit:JUnit是一個優(yōu)秀的單元測試框架,它利用了 Java反射機制動態(tài)地加載和執(zhí)行測試方法。
  • Jackson:Jackson是一個 JSON處理的 Java庫,它利用反射來實現(xiàn) JSON與 Java對象之間的轉(zhuǎn)換,動態(tài)讀取和寫入 Java對象的屬性,并將其轉(zhuǎn)換為 JSON格式。
  • Hibernate ORM:Hibernate和 MyBatis一樣,都是對象關系映射框架,通過反射來實現(xiàn)對象與數(shù)據(jù)庫表之間的映射關系。

八、總結 

本文講解了Java反射的原理和使用方式,因為有了Java反射,很多優(yōu)秀的框架應運而生,從而使得 Java 生態(tài)越來越完善,因此,反射是絕大多數(shù)框架的基石。

Java反射有優(yōu)點也有缺點,從整體上看,Java反射是以犧牲了小部分的性能換取了更好的擴展性和靈活性,犧牲小我成就大我,而且,隨著現(xiàn)代硬件設備能力越來越強,這點小性能的犧牲是完全值得的。

掌握Java反射,我們可以更好的理解一些優(yōu)秀框架的運行機制,比如:Spring。它可以幫助我們更好的使用框架,遇到問題時也能更好的去分析和解決。

責任編輯:趙寧寧 來源: 猿java
相關推薦

2020-04-03 14:55:39

Python 代碼編程

2022-03-08 06:41:35

css代碼

2019-09-29 16:17:25

Java代碼性能編程語言

2022-04-10 10:41:17

ESLint異步代碼

2023-11-23 13:50:00

Python代碼

2022-11-18 08:32:23

spring參數(shù)解析器

2018-07-12 14:20:33

SQLSQL查詢編寫

2022-12-26 07:47:37

JDK8函數(shù)式接口

2023-07-10 09:39:02

lambdaPython語言

2025-02-10 00:25:00

命令模式擴展機制系統(tǒng)

2025-05-29 08:35:39

Python代碼開發(fā)

2025-04-21 17:55:25

2017-09-27 16:09:29

代碼

2024-12-31 08:54:38

2023-12-21 10:26:30

??Prettier

2022-05-13 08:48:50

React組件TypeScrip

2024-07-03 08:13:56

規(guī)則執(zhí)行器代碼

2024-08-20 14:25:20

2024-07-30 14:09:19

裝飾器Python代碼

2021-06-25 15:53:25

Kubernetes程序技巧
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产目拍亚洲精品99久久精品 | 日韩在线观看一区 | 天天影视色综合 | 欧美精品一区二区三区在线播放 | 国产成都精品91一区二区三 | 国产亚洲欧美在线 | 成人中文字幕在线观看 | 欧美成视频 | 久久精品免费 | 国产a区 | 中文字幕91| 成人免费在线网 | 久久国产精品免费视频 | 日韩一区在线观看视频 | 国产精品一区在线 | 国产精品久久一区二区三区 | 精品国产一区二区 | 久久国产欧美日韩精品 | 操人网 | 欧美在线观看一区 | 国产精品久久久久久久久久 | 中文字幕在线观看 | 久久久人成影片免费观看 | 中文字幕高清 | 国产精品国产自产拍高清 | 99久久国产综合精品麻豆 | 国产精品久久久久久久久 | 日韩在线国产 | 久久久精品日本 | 亚洲一区二区三区视频 | 四虎永久在线精品免费一区二 | 7777精品伊人久久精品影视 | 国产欧美精品 | 久久久久亚洲精品中文字幕 | 99精品视频在线观看 | 国产成人a亚洲精品 | 亚洲国产成人精品一区二区 | 91精品国产91久久久久久吃药 | 免费在线一区二区 | 国产精品久久久久久久久图文区 | 亚洲精品美女视频 |