最近一个项目有一个需求,需要去获取方法的参数名,我们知道,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自动添加了这些参数,不信你打开设置看一下:
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选项
欢迎扫码查看更过文章: