欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java通過反射獲取方法參數(shù)名的方式小結(jié)

 更新時間:2025年02月11日 15:32:15   作者:李白的手機(jī)  
這篇文章主要為大家詳細(xì)介紹了Java如何通過反射獲取方法參數(shù)名的方式,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下

1、前言

一般當(dāng)我們用類似下面的反射去獲取方法參數(shù)名時得到的一般是 arg0、arg1, 參數(shù)名默認(rèn)會丟失, 導(dǎo)致無法獲取到真實(shí)的方法參數(shù)名,下面介紹如何通過其他方式解決。

    class A {
        void getUser(String userName, String userId){}
    }
    
    Method method = A.class.getMethod(String.class,String.class);
    Parameter[] parameters = method.getParameters();
    for (Parameter parameter : parameters) {
        String name = parameter.getName();
        System.out.println(name); // 得到的是arg0, arg1 而非 userName、userId
    }

2、解決方式

方式2.1: 添加編譯參數(shù)配置 -parameters

java默認(rèn)在編譯工程代碼為class文件時,會將方法參數(shù)名擦除,替換成arg0、arg1之類。但是我們可以在編譯時配置將參數(shù)名也一并編譯進(jìn)去,在maven中新增如下編譯插件配置

            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <configuration>
                    <!-- 配置編譯時將方法參數(shù)名保留 -->
                    <compilerArgs>
                        <arg>-parameters</arg>
                    </compilerArgs>
                </configuration>
            </plugin>

對于較新的 maven 版本(>= 3.6.2), 也可以直接使用 parameters 配置項(xiàng):

<plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-compiler-plugin</artifactId>
    <configuration>
        <parameters>true</parameters>
    </configuration>
</plugin>

添加-parameters的編譯參數(shù)后,可以看到我們編譯后的class文件是源碼的參數(shù)名,而非原來的arg0、arg1。 之后我們用原生的反射Parameter.getName() 就可以獲取到的方法參數(shù)的真實(shí)參數(shù)名了

注意: 該方式僅對JDK8及以上版本有效, 以前版本JDK沒有提供該保留機(jī)制

方式2.2: 使用Spring的內(nèi)部工具類 - ParameterNameDiscoverer

如果正好你的項(xiàng)目依賴了spring,則可以直接使用spring的內(nèi)部工具類ParameterNameDiscoverer去獲取到方法真實(shí)參數(shù)名。ParameterNameDiscoverer的實(shí)現(xiàn)類 主要包含 LocalVariableTableParameterNameDiscoverer、StandardReflectionParameterNameDiscoverer 。 下面介紹如何使用

1、StandardReflectionParameterNameDiscoverer的使用

該發(fā)現(xiàn)器需要與 方式2.1: 添加編譯參數(shù)配置 -parameters 一樣添加編譯配置, 因?yàn)榈讓右彩侵苯佑梅瓷銹arameter去獲取而已

  // 創(chuàng)建參數(shù)名發(fā)現(xiàn)器
  ParameterNameDiscoverer discoverer = new StandardReflectionParameterNameDiscoverer();

  Method method = A.class.getMethod(String.class,String.class);

  // 獲取方法真實(shí)參數(shù)名.  userName, userId
  String[] parameterNames = discoverer.getParameterNames(method);

2、LocalVariableTableParameterNameDiscoverer的使用

  // 創(chuàng)建參數(shù)名發(fā)現(xiàn)器
  ParameterNameDiscoverer discoverer = new LocalVariableTableParameterNameDiscoverer();

  Method method = A.class.getMethod(String.class,String.class);

  // 獲取方法真實(shí)參數(shù)名.  userName, userId
  String[] parameterNames = discoverer.getParameterNames(method);

該發(fā)現(xiàn)器底層原理是在編譯類時生成一個調(diào)試信息的局部變量表LocalVariableTable,然后基于ASM字節(jié)碼技術(shù)去解析LocalVariableTable得到參數(shù)名, 所以它也需要在編譯時額外添加 編譯配置, 請在maven中添加如下-g的編譯參數(shù)配置

    <build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <configuration>
                    <compilerArgs>
                        <!-- 生成局部變量表 -->
                        <arg>-g:vars</arg>
                    </compilerArgs>
                </configuration>
            </plugin>
        </plugins>
    </build>

該方法優(yōu)點(diǎn)是JDK8以下版本也能使用,但是缺點(diǎn)是無法獲取到接口的方法參數(shù)名

3、DefaultParameterNameDiscoverer的使用

  // 創(chuàng)建參數(shù)名發(fā)現(xiàn)器
  ParameterNameDiscoverer discoverer = new DefaultParameterNameDiscoverer();

  Method method = A.class.getMethod(String.class,String.class);

  // 獲取方法真實(shí)參數(shù)名.  userName, userId
  String[] parameterNames = discoverer.getParameterNames(method);

這個參數(shù)名發(fā)現(xiàn)器是一個組合模式的發(fā)現(xiàn)器, 本身不去實(shí)現(xiàn)獲取參數(shù)名的邏輯,而是組合其他參數(shù)名發(fā)現(xiàn)器, 然后迭代調(diào)用每個參數(shù)名發(fā)現(xiàn)器直到找到真實(shí)參數(shù)名。 從下面源碼來看 主要內(nèi)置了前面講到的兩種StandardReflectionParameterNameDiscoverer 和 LocalVariableTableParameterNameDiscoverer。 所以如果要讓對應(yīng)的參數(shù)名發(fā)現(xiàn)器生效,需要配合對應(yīng)發(fā)現(xiàn)器的使用邏輯 否則一樣無法獲取到方法參數(shù)名。

// 源碼
public class DefaultParameterNameDiscoverer extends PrioritizedParameterNameDiscoverer {

	public DefaultParameterNameDiscoverer() {
		if (!GraalDetector.inImageCode()) {
			if (KotlinDetector.isKotlinReflectPresent()) {
				addDiscoverer(new KotlinReflectionParameterNameDiscoverer());
			}
			addDiscoverer(new StandardReflectionParameterNameDiscoverer()); // 依賴編譯參數(shù) -parameters
			addDiscoverer(new LocalVariableTableParameterNameDiscoverer()); //  // 依賴編譯參數(shù) -g
		}
	}

}

使用總結(jié)

1、參數(shù)名獲取的本質(zhì)

  • 所有方案都依賴編譯時保留參數(shù)名信息(通過-parameters或-g參數(shù)), 所以編譯的時候有就是有,沒有就是沒有,尤其是一些第三方依賴的類
  • 如果是第三方依賴的類若未攜帶調(diào)試信息(如未使用上述編譯參數(shù)進(jìn)行編譯),則無論如何也是無法獲取真實(shí)參數(shù)名, 除非第三方庫已使用-parameters編譯,仍可通過反射獲取(但實(shí)際場景較少見)

2、版本適配建議

  • JDK8+項(xiàng)目: 優(yōu)先使用-parameters編譯參數(shù)(性能最佳), 配合StandardReflectionParameterNameDiscoverer
  • JDK7及以下: 使用-g編譯參數(shù), 配合LocalVariableTableParameterNameDiscoverer
  • Spring項(xiàng)目: 推薦使用DefaultParameterNameDiscoverer(自動適配最優(yōu)策略), 可同時配置兩種編譯參數(shù)提升兼容性:

3、在一些springBoot 2.0以后版本中spring-boot-starter-parent會內(nèi)置了編譯參數(shù)配置,所以不用配也可以直接使用ParameterNameDiscoverer, 當(dāng)然需要確保項(xiàng)目是否覆蓋了默認(rèn)配置導(dǎo)致失效,需要重新配

知名應(yīng)用場景舉例

  • spring-mvc 的 @RequestParam 參數(shù)名的可省略配置
  • mybatis 的 @Param 也可以不用配置

到此這篇關(guān)于Java通過反射獲取方法參數(shù)名的方式小結(jié)的文章就介紹到這了,更多相關(guān)Java反射獲取方法參數(shù)名內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • mybatis-plus中更新null值的問題解決

    mybatis-plus中更新null值的問題解決

    本文主要介紹 mybatis-plus 中常使用的 update 相關(guān)方法的區(qū)別,以及更新 null 的方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2024-04-04
  • IntelliJ IDEA查看方法說明文檔的圖解

    IntelliJ IDEA查看方法說明文檔的圖解

    今天小編就為大家分享一篇關(guān)于IntelliJ IDEA查看方法說明文檔的圖解,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧
    2018-10-10
  • 在啟動后臺 jar包時,使用指定的 application.yml操作

    在啟動后臺 jar包時,使用指定的 application.yml操作

    這篇文章主要介紹了在啟動后臺 jar包時,使用指定的 application.yml操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-10-10
  • 使用OpenFeign實(shí)現(xiàn)服務(wù)調(diào)用的方法詳解

    使用OpenFeign實(shí)現(xiàn)服務(wù)調(diào)用的方法詳解

    OpenFeign是運(yùn)行在客戶端的聲明式服務(wù)調(diào)用的框架,通過聲明接口的方式來達(dá)到對服務(wù)的調(diào)用,本文就來和大家講講如何使用OpenFeign實(shí)現(xiàn)服務(wù)調(diào)用吧
    2023-06-06
  • java通過釘釘機(jī)器人發(fā)消息的實(shí)現(xiàn)示例

    java通過釘釘機(jī)器人發(fā)消息的實(shí)現(xiàn)示例

    本文主要介紹了java通過釘釘機(jī)器人發(fā)消息的實(shí)現(xiàn)示例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-09-09
  • 深入剖析Java中String類的concat方法

    深入剖析Java中String類的concat方法

    這篇文章主要介紹了Java中String類的concat方法,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-12-12
  • Springboot基于assembly的服務(wù)化打包方案及spring boot部署方式

    Springboot基于assembly的服務(wù)化打包方案及spring boot部署方式

    這篇文章主要介紹了Springboot基于assembly的服務(wù)化打包方案及springboot項(xiàng)目的幾種常見的部署方式,本文主要針對第二種部署方式提供一種更加友好的打包方案,需要的朋友可以參考下
    2017-12-12
  • 使用SpringBoot開發(fā)Restful服務(wù)實(shí)現(xiàn)增刪改查功能

    使用SpringBoot開發(fā)Restful服務(wù)實(shí)現(xiàn)增刪改查功能

    Spring Boot是由Pivotal團(tuán)隊(duì)提供的全新框架,其設(shè)計目的是用來簡化新Spring應(yīng)用的初始搭建以及開發(fā)過程。這篇文章主要介紹了基于SpringBoot開發(fā)一個Restful服務(wù),實(shí)現(xiàn)增刪改查功能,需要的朋友可以參考下
    2018-01-01
  • java處理數(shù)據(jù)庫不支持的emoji表情符問題解決

    java處理數(shù)據(jù)庫不支持的emoji表情符問題解決

    這篇文章主要介紹了java處理數(shù)據(jù)庫不支持的emoji表情符問題解決,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2019-09-09
  • Java中ArrayList和Vector的區(qū)別

    Java中ArrayList和Vector的區(qū)別

    本文主要介紹了Java中ArrayList和Vector的區(qū)別,包括線程安全性、性能、同步機(jī)制、擴(kuò)容機(jī)制、遍歷方式等,具有一定的參考價值,感興趣的可以了解一下
    2024-11-11

最新評論