今天发生了一件事,令我非常郁闷,就是我在使用一个SDK时,当我调用他的方法时,提示我方法中的参数var1, var2如下:
// 方法是我随意构造的,不代表真实方法
excuseMe(String var1, String var2);
当时我就想对提供SDK的这位仁兄说一句,年轻人不讲武德,我劝你耗子尾汁。
对于这样的API,真的我是看都不想看,你给我整个var0,var1我哪知道这参数到底代表啥,难道我们之间还有摩斯密码,我需要去解密一下,兄台大家都是开发,何苦自相残杀。
既然我已经受到了这段代码的困扰,那么不如我们研究一下为什么会有这种情况发生。
这种情况发生的原因其实是因为Class文件中方法表的Code属性中有一个属性没生成,这个属性是LocalVariableTable。
LocalVariableTable用来描述栈帧中的局部变量表的变量和与Java源码中定义的变量之间的关系,其结构如下:
文章图片
关于attribute_name_index和attribute_length的含义如果还有不懂的读者可以建议将我之前讲解的Class文件相关文章阅读一下。
- local_variable_table_length:local_variable_info的个数
- local_variable_info:一个栈帧与源码中的局部变量的关联关系
文章图片
- start_pc:局部变量开始的字节码偏移量
- length:局部变量作用范围的覆盖长度
- name_index:局部变量名称(CONSTANT_Utf8_info)在常量池中的索引
- descriptor_index:局部变量的(CONSTANT_Utf8_info)描述在常量池中的索引
- index:局部变量在栈帧的局部变量表中的Slot的位置,如果数据类型是64位,那么他所占用的Slot是index和index+1
public class ClassTest {private Integer number;
public void excuseMe(String from, String to) {
number = 1;
System.out.println("excuse me");
}}
我们可以通过上述这段代码来研究一下我们开篇的问题:
javac ClassTest.java
我们看一下编译出来的Class文件,execuseMe方法里面的参数为var1,var2,出现了我们开篇的问题
文章图片
那么我们怎么解决这个问题,其实只需要在编译的时候加入一个参数-g:vars
javac -g:vars ClassTest.java
可以看出加入这个参数后,Class文件中的变量已经具有含义了from,to
文章图片
我们再通过一下命令看一下LocalVariableTable这个属性有没有生成
javap -v ClassTest
可以看到LocalVariableTable属性已经生成到excuseMe方法的Code属性中。
文章图片
我是shysh95,最好的开始是十年前,其次是现在,希望可以和你专注技术的路上并肩作战,我们下期再见!!!
【这样的SDK不讲武德】扫码关注微信公众号,更多精彩文章!!!
文章图片