add framework support中没有maven_Java中到底能不能在运行时获取方法的参数名?why?how?...

最近一个项目有一个需求,需要去获取方法的参数名,我们知道,java的源文件首先是编译成class文件,jvm在运行时是执行的class文件的字节码, 那么,如果想获取到方法的参数名,首先要保证class文件中得有参数的名字才可以,那么我们就来看一下,默认的javac编译出来的class文件中是否是带有参数名的:

//这是一个非常简单的类:
package com.github.xjs;
public class CompilerDemo {
    public static void main(String[] args) {
        System.out.println("compiler demo");
    }
    public static int add(int a,int b){
        return a+b;
    }
}

使用javac来编译一下:

cd E:workspacecompilerdemosrcmainjavacomgithubxjs
//编译
javac CompilerDemo.java
//打印字节码
javap -verbose CompilerDemo.class

截取add(int a,int b)这个方法的字节码看下:

public static int add(int, int);
    descriptor: (II)I
    flags: ACC_PUBLIC, ACC_STATIC
    Code:
      stack=2, locals=2, args_size=2
         0: iload_0
         1: iload_1
         2: iadd
         3: ireturn
      LineNumberTable:
        line 8: 0

这里面只有行号表LineNumberTable,并没有参数的任何信息,也就是说如果是拿着这样的字节码,就算是神仙也是没办法得到方法的参数名的。

但是,javac有一个编译选项-g,具体可以看下这里:https://docs.oracle.com/javase/8/docs/technotes/tools/unix/javac.html

-g: 用来在class文件中产生调试信息,包括了:本地变量,如果不加,默认只有行号和源码的信息。
-g:none:不产生任何的调试信息
-g:[keyword list]:可以有多个关键字,以逗号分隔,用于产生不同的调试信息,source:源码的调试信息,lines:行号的调试信息,vars:本地变量的调试信息

我们加一下-g来试试效果:

javac  -g CompilerDemo.java
javap -verbose CompilerDemo.class

重新看下add(int a,int b)这个方法的字节码:

public static int add(int, int);
    descriptor: (II)I
    flags: ACC_PUBLIC, ACC_STATIC
    Code:
      stack=2, locals=2, args_size=2
         0: iload_0
         1: iload_1
         2: iadd
         3: ireturn
      LineNumberTable:
        line 8: 0
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       4     0     a   I
            0       4     1     b   I

可以看到多出来了一个LocalVariableTable,也就是本地变量表,这里面就包含了方法的参数信息。

同时呢,jdk8还提供了一个新的编译选项和新的获取参数的api,这个选项是:-parameters.

我们用这个选项来试一下:

//javac  -parameters CompilerDemo.java
//javap -verbose CompilerDemo.class
public static int add(int, int);
    descriptor: (II)I
    flags: ACC_PUBLIC, ACC_STATIC
    Code:
      stack=2, locals=2, args_size=2
         0: iload_0
         1: iload_1
         2: iadd
         3: ireturn
      LineNumberTable:
        line 8: 0
    MethodParameters:
      Name                           Flags
      a
      b

可以看到,添加了这个选项以后,生成的字节码多了一个叫做MethodParameters的东西,这里面也有参数,并且呢,jdk8还提供了相应的api来获取这个参数的名字,看文档:

-parameters:在class文件中存储构造函数和方法的参数名,然后可以用java.lang.reflect.Executable.getParameters的api来获取参数名。

我们都知道,Spring在做Controller方法的参数绑定的时候,可以这样:

@GetMapping("/demo")
public String demo(String a, String b){
  return a + "," + b;
}

然后浏览器访问:http://localhost:8989/demo?a=1&b=2这样是可以做参数绑定的,那么Spring背后是如何来实现的呢?

Spring参数绑定最终是使用这个类:org.springframework.core.DefaultParameterNameDiscoverer:

public DefaultParameterNameDiscoverer() {
  if (!GraalDetector.inImageCode()) {
    if (KotlinDetector.isKotlinReflectPresent()) {
      addDiscoverer(new KotlinReflectionParameterNameDiscoverer());
    }
    //这个是使用jdk8的Parameter的api来获取,也就是说编译开启了-parameters的时候
    addDiscoverer(new StandardReflectionParameterNameDiscoverer());
    //这个是使用ASM来访问class字节码里面的本地变量表,也就是编译开启了-g的时候
    addDiscoverer(new LocalVariableTableParameterNameDiscoverer());
  }
}

上面可以看出来,Spring最终还是使用的-parameters或者-g,如果没有这两个选项,Spring也是无能为力的,如果恰好真的就没开启这两个选项,在Spring中可以用@RequestParam手动明确指定:

@GetMapping("/demo")
public String demo(@RequestParam("a") String a, @RequestParam("b")String b){
  return a + "," + b;
}

可能很多同学会有疑问,我们在用idea开发的时候,没有去开启-parameters和-g啊,为啥也能参数绑定?原因是idea自动添加了这些参数,不信你打开设置看一下:

d0b7fe3beee25af9b4678705d9c46700.png

idea可以说是真的很贴心了。

还有同学会有疑问,我们开发用的idea,但是打包是用的maven,难道maven在编译的时候也开启了这两个选项?

看下maven-compiler-plugin的文档:http://maven.apache.org/plugins/maven-compiler-plugin/compile-mojo.html

注意下插件的这几个参数:

- 1.<debug>:默认值是true,可以把调试信息添加到class文件中
- 2.<debuglevel>:定义了出现在-g参数后面的选项,要么啥都不填,要么是以后号分隔的lines, vars, source,如果不设置,默认就是-g,不带其他选项。只有在开启debug有效。

因此,可以看出来,默认情况下,maven-compiler-plugin实际上就是给javac添加了-g参数,也就是如下:

<plugins>
  <plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-compiler-plugin</artifactId>
    <version>3.8.0</version>
    <configuration>
      <debug>true</debug>
      <compilerArgument>-g</compilerArgument>
    </configuration>
  </plugin>
</plugins>

总结一下
- 1.要想在运行时获取方法的参数名,首先class文件中得有参数名才可以
- 2.javac可以通过添加-g或者-parameter选项在class文件中添加参数信息
- 3.idea默认是添加了-parameter选项
- 4.maven默认是添加了-g选项

欢迎扫码查看更过文章:

db3967bc660044981904f745f0966ead.png
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值