# Java8 获取参数名及 Idea/Eclipse/Maven 配置
在 Java8 之前,代码编译为 class 文件后,方法参数的类型固定,但是方法名称会丢失,方法名称会变成 arg0、arg1....。而现在,在 Java8 开始可以在 class 文件中保留参数名,这就给反射带来了极大的遍历。像 mybatis 等需要使用反射机制获取方法参数的时候就可以不用像以前一样需要使用类似于
@Param之类的注解。
# 功能测试
- 编写测试类
import java.lang.reflect.Method;
import java.lang.reflect.Parameter;
public class GetRuntimeParameterName {
public void createUser(String name, int age, int version) {
}
public static void main(String[] args) throws Exception {
for (Method m : GetRuntimeParameterName.class.getMethods()) {
System.out.println("--------------------");
System.out.println(" method: " + m.getName());
System.out.println(" return: " + m.getReturnType().getName());
for (Parameter p : m.getParameters()) {
System.out.println("parameter:" + p.getType().getName() + ", " + p.getName());
}
}
}
}
- 测试
由于为了避免.class 文件因为保留参数名而导致.class 文件过大或者占用更多的内存,另外也避免有些参数(secrect/password)泄露安全信息,JVM 即使时 1.8 默认是不会保留参数名称的。
所以我们这里正常可以测试编译保留参数名和不保留参数名的情况。
- 不保留参数名称 编译命令:
javac GetRuntimeParameterName.java
输出结果:
method: createUser
return: void
parameter:java.lang.String, arg0
parameter:int, arg1
parameter:int, arg2
保留参数名
编译命令:
javac -parameters GetRuntimeParameterName.java
输出结果:
--------------------
method: createUser
return: void
parameter:java.lang.String, name
parameter:int, age
parameter:int, version
# IDE 和 Maven 开启-parameters 的办法
# Eclipse 中开启的办法
Preferences->java->Compiler下勾选Store information about method parameters选项。
这样在使用 eclipse 编译 java 文件的时候就会将参数名称编译到 class 文件中。
# Idea 中开启的方法
File->Settings->Build,Execution,Deployment->Java Compiler下的Additional command line parameters选项中添加-parameters。

# Maven 中开启的办法
在 pom.xml 的编译插件中增加参数配置<arg>-parameters</arg>。
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.3</version>
<configuration>
<source>1.8</source>
<target>1.8</target>
<compilerArgs>
<arg>-parameters</arg>
</compilerArgs>
</configuration>
</plugin>
另外附上测试使用 Maven install 命令编译 class 文件的情况:
使用 maven install 编译生成 jar 包时:
- 如果 target/classes 下 class 文件对于源码文件都没有修改,不会重新编译,直接打包 classes 下的 class 文件;
- 如果 target/classes 下 class 文件对于源码文件都有修改,会重新编译所有 class 文件,然后打包 classes 下的 class 文件;