最近一个项目有一个需求,需要去获取方法的参数名,我们知道,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:\workspace\compilerdemo\src\main\java\com\github\xjs //编译 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.:默认值是true,可以把调试信息添加到class文件中2.:定义了出现在-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-compiler-plugin默认是添加了-g选项欢迎扫码加关注查看更多文章: