`

JAVA获取方法参数名的分析(二)

    博客分类:
  • Java
阅读更多

2010年5月13日 魏超

 

上一节:JAVA获取方法参数名的分析(一)

 

javac和Eclipse编译出来的class文件在传入参数名称上的区别

在上文中可以看出,由javac编译的类,IDE在引用了改类的时候,无法获得方法传入参数原来的命名,只能重新赋予arg0,arg1之类的名字。而由eclipse(MyEclipse)编译出来的class文件,却具有传入参数原来的名字。

 

为了能进一步清楚这个区别,我们这里借助一个软件来查看class文件:JClassLib

JClassLib 写道
JClassLib不但是一个字节码阅读器而且还包含一个类库允许开发者读取,修改,写入Java Class文件与字节码。
 

我这里只使用了jclasslib的GUI那个版本。jclasslib作为一个java库还可以有更多更强大的功能。用jclasslib分别打开2个class文件,如下图:


上图可以看出,2个class文件相同的testJar方法下,下面那个testJar方法中还注入了"[1]LocalVariableTable"属性。这个属性通过右边的详细可以看出,存储的就是jarName和yourName2个传入参数的名字。而上面那个javac编译出来的class是没有这个属性的。

对于没有注入参数名称的class,按照我的推断,他应该是根据


上面这个方法中的Descriptor这个标志来判断传入参数是什么类型的,再根据类型给一个默认的名称。

 

注:上图中用红色框起来的descriptor的意思是  <(参数1数据类型,参数2数据类型) 方法返回类型>

 

因此,我们使用Eclipse等IDE在调用spring.jar等jar的方法时,代码提示可以正确读取到传入参数的原命名是因为 spring.jar等jar包在编译时就向class文件注入了参数名字。因此,eclipse等IDE也是通过读取.class的字节码来获得方法的参数名称的。

 

通过网上资料得知,eclipse\myEclipse并没有引用sun默认的javac编译器,而是有自己的实现,因此在编译java文件的时候,由eclipse编译出来的与由javac编译出来的就有所不同。

 

 

Eclipse/MyEclipse中 .class没有注入参数名称的jar在 attach source之后的区别

上一节的分析,我们了解到,倘若在生成.class文件的时候没有注入参数名称的信息,那么在引入这个class之后,使用它类的方法时,代码提示的参数就只会是arg0,arg1之类的名称。但当我们对这个引入的jar包进行了  attach source之后,代码提示的时候就可以按照source中的正确参数名称来显示。

是不是attach source之后,IDE向jar包中写入了参数名称的信息?

用jclasslib打开attach source以后的class.可以发现这个.class没有发生变化。那么还有一种可能性,就是进行了这个操作之后,eclipse再进行代码提示的时候,是直接从source中读取参数名称的。为了验证这点,我们修改source中方法的参数名。将jarName改成jarNameTest. 再次使用代码提示,可以看到,原来的参数名字没有发生变化。因此,IDE并没有每次都从sourc中读取参数的名称。而是在attach的时候读取了一次,之后做了缓存。

 

看到这里,我的第一个反应是eclipse在这个工程的目录下保存了类似“索引”的东西,将方法参数对应的名称保存了下来。针对这个猜测,我监测了attach前后的工程目录情况,如下图:



 发生变化的只有.classpath文件。source的路径被写到了这个文件中。因此,IDE并没有建立索引。那么可以猜测,eclipse可能在内存中做了缓存,也可能是在自己的公共目录下建立了索引。

 

对于在内存中缓存的可能,我们只要杀死eclipse的进程便可以验证(重启eclipse).结果发现,参数名称变成了修改后的。那么可以看出,这个参数名称的缓存,eclipse是放在自己开辟的一块内存空间内的,只是临时的。在失去了这个缓存之后就会重新读取。

 

 

小结

通过上面的分析,我们可以得出一个结论,eclipse等IDE在代码提示时对方法参数名称的获取,有2种途径。

A.当.class文件中持有 参数名称 信息的时候,会直接从class中读取。

B.当没有时,从.java源文件中,通过字节流读取参数名称,然后缓存起来。

 

因此,我们要模仿IDE的做法,获得一个方法的传入参数名称,我们有上面2种途径。具体的实现,在下一篇心得里会继续分析。

  • 大小: 96.4 KB
  • 大小: 35.2 KB
  • 大小: 52.4 KB
分享到:
评论
4 楼 ocaicai 2013-12-13  
https://gist.github.com/wendal/2011728#file-methodparamnamesscaner-java
3 楼 ta8210 2010-11-11  
我这里有2种办法供你选择
1.使用javassist.jar 这个项目可以帮助你解决这个问题,它是一个增强的反射工具。
项目地址http://www.csg.is.titech.ac.jp/~chiba/javassist/
使用例子http://www.blogjava.net/Hafeyang/archive/2010/10/25/using_powerful_java_reflect_tool_javassist_to_getParameterAnnotations_and_getParameterNames.html
但是这个工具的运行效率没有ASM高。

你也可以使用ASM来扫描Class文件这样获取你要的属性名,前提是你需要对 Class字节码有比较深入的了解。

我更趋向于使用后者,但是ASM的API没有javassist方便。
2 楼 javaest 2010-07-09  
还要说一句,很期待你的本系列的第三篇文章.
1 楼 javaest 2010-07-09  
是个细心人 ,是一片好文章.
我也关注这个问题.
在我的Web框架中,我需要根据参数名称自动到Request中的parameter中取得参数值.
最后没办法我在每个参数上加了标注信息解决这个问题.
你也可以借鉴一下.

    public JSONObject updateFunctionMenuTree(@Para("moduleId") String moduleId ,@Para("roleIdCommaStrForMenu") String roleIdCommaStr,@Para("state") String state) {
        System.out.println(state);
        List<String> menuIdList=menuAndFunctionDao.getAllMenuId(moduleId);
        String[] roleIdArr=roleIdCommaStr.split(",");
        System.out.println(roleIdArr.length);
        int addItemCount=0;
        for (int i = 0; i < roleIdArr.length; i++) {
            addItemCount=securityManager.tempAddSecurityMenuIdForRole(roleIdArr[i], menuIdList);
        }        
        consoleTreeMenuBuilder.reloadTreeMenu(moduleId);
        JSONObject object=new JSONObject();
        object.put("msg", "成功更新功能菜单树 ! "+(addItemCount==0?"":"\n并为选中角色临时添加了 "+addItemCount+" 项菜单的操作权限 ! "));
        return object;
    }


在JAVA 7.0 中,编译时就不会丢掉参数名称信息了.据说.这个说法源于:http://www.ibm.com/developerworks/cn/java/j-javaroundtable/index.html

相关推荐

    Java实现微信公众号获取临时二维码功能示例

    主要介绍了Java实现微信公众号获取临时二维码功能,结合实例形式分析了java调用微信公众号接口实现临时二维码生成功能相关操作技巧,需要的朋友可以参考下

    php获取不到自定义header参数原因详解

    详细分析php获取不到自定义header参数的原因,明确设置自定义header参数注意事项以及php获取自定义参数方法

    java源码包---java 源码 大量 实例

     Java波浪文字,一个利用Java处理字符的实例,可以设置运动方向参数,显示文本的字符数组,高速文本颜色,显示字体的 FontMetrics对象,得到Graphics实例,得到Image实例,填充颜色数组数据,初始化颜色数组。...

    Java 模拟cookie登陆简单操作示例

    主要介绍了Java 模拟cookie登陆简单操作,结合实例形式分析了Java 模拟cookie登陆的相关原理与基本实现技巧,需要的朋友可以参考下

    125集专攻JAVA基础 JAVA零基础入门学习视频教程 动力节点JAVA视频教程.txt

    北京动力节点-Java编程零基础教程-115-Java基本语法-方法详解-方法的参数与返回值的应用场景.avi 北京动力节点-Java编程零基础教程-116-Java基本语法-方法详解-变量的作用域.avi 北京动力节点-Java编程零基础教程...

    JAVA上百实例源码以及开源项目

     Java波浪文字,一个利用Java处理字符的实例,可以设置运动方向参数,显示文本的字符数组,高速文本颜色,显示字体的 FontMetrics对象,得到Graphics实例,得到Image实例,填充颜色数组数据,初始化颜色数组。...

    JAVA上百实例源码以及开源项目源代码

     Java波浪文字,一个利用Java处理字符的实例,可以设置运动方向参数,显示文本的字符数组,高速文本颜色,显示字体的 FontMetrics对象,得到Graphics实例,得到Image实例,填充颜色数组数据,初始化颜色数组。...

    Java调用weka神经网络算法预测股票 代码及数据

    Java调用weka神经网络算法预测股票,含有代码及数据,代码有详细解释,数据为大量数据。

    java源码包4

     Java波浪文字,一个利用Java处理字符的实例,可以设置运动方向参数,显示文本的字符数组,高速文本颜色,显示字体的 FontMetrics对象,得到Graphics实例,得到Image实例,填充颜色数组数据,初始化颜色数组。...

    java源码包3

     Java波浪文字,一个利用Java处理字符的实例,可以设置运动方向参数,显示文本的字符数组,高速文本颜色,显示字体的 FontMetrics对象,得到Graphics实例,得到Image实例,填充颜色数组数据,初始化颜色数组。...

    Java 获取resource的源码分析

    java 中获取 resource一般通过getResource(),不管你是通过class还是classloader来调用, 首先我们来讲讲getResource(string name)这个name参数,通过class调用getResource(string name)方法, name会先调用resolve...

    java开源包8

    Blister是一个用于操作苹果二进制PList文件格式的Java开源类库(可用于发送数据给iOS应用程序)。 重复文件检查工具 FindDup.tar FindDup 是一个简单易用的工具,用来检查计算机上重复的文件。 OpenID的Java客户端...

    JAVA入门1.2.3:一个老鸟的JAVA学习心得 PART1(共3个)

    12.1.3 仔细分析recordTransport()方法 338 12.2 初用接口 339 12.2.1 准备好需要用到的类 339 12.2.2 认识接口的代码组成 340 12.2.3 什么是接口 341 12.2.4 使用接口仅需一步——实现接口 342 12.2.5 接口...

    java源码包2

     Java波浪文字,一个利用Java处理字符的实例,可以设置运动方向参数,显示文本的字符数组,高速文本颜色,显示字体的 FontMetrics对象,得到Graphics实例,得到Image实例,填充颜色数组数据,初始化颜色数组。...

    Java入门1·2·3:一个老鸟的Java学习心得.PART3(共3个)

    12.1.3 仔细分析recordTransport()方法 338 12.2 初用接口 339 12.2.1 准备好需要用到的类 339 12.2.2 认识接口的代码组成 340 12.2.3 什么是接口 341 12.2.4 使用接口仅需一步——实现接口 342 12.2.5 接口...

    java 面试题 总结

    redirect就是服务端根据逻辑,发送一个状态码,告诉浏览器重新去请求那个地址,一般来说浏览器会用刚才请求的所有参数重新请求,所以session,request参数都可以获取。 20、EJB与JAVA BEAN的区别? Java Bean 是可复用...

    java开源包11

    Blister是一个用于操作苹果二进制PList文件格式的Java开源类库(可用于发送数据给iOS应用程序)。 重复文件检查工具 FindDup.tar FindDup 是一个简单易用的工具,用来检查计算机上重复的文件。 OpenID的Java客户端...

    java开源包6

    Blister是一个用于操作苹果二进制PList文件格式的Java开源类库(可用于发送数据给iOS应用程序)。 重复文件检查工具 FindDup.tar FindDup 是一个简单易用的工具,用来检查计算机上重复的文件。 OpenID的Java客户端...

    java开源包9

    Blister是一个用于操作苹果二进制PList文件格式的Java开源类库(可用于发送数据给iOS应用程序)。 重复文件检查工具 FindDup.tar FindDup 是一个简单易用的工具,用来检查计算机上重复的文件。 OpenID的Java客户端...

    java开源包101

    Blister是一个用于操作苹果二进制PList文件格式的Java开源类库(可用于发送数据给iOS应用程序)。 重复文件检查工具 FindDup.tar FindDup 是一个简单易用的工具,用来检查计算机上重复的文件。 OpenID的Java客户端...

Global site tag (gtag.js) - Google Analytics