各位朋友,上一篇的常量池都掌握了吧,我们的口号是:每天进步一点点!
开始之前,我们先巩固并扩充一下昨天的知识,用Excel表示一下就是这样的(请原谅笔者low到爆的画图水平,画这种图有什么好的工具,朋友们可以推荐一下啊)。
请注意!!!这里有个地方需要补充个一下,那就是当常量的类型是CONSTANT_Utf8_info的时候,看一下它的类型,length是它的字符串长度,意思是代表有多少个bytes(字节)。
我们举一个例子,tag为“01”,接着的“00
06”说明长度为6,向后查四个字节,分别为“3c”、“69”、“6e”、“69”、“74”、“3e”,转换成十进制为60、105、110、105、116和62,既然是字符串,我们对照一下ASCII码,居然是“”!!!我得天哪!
眼见为实(密集恐惧症不要看!)。
下面开始今天的学习。
五、访问标志:对照一下结构表,在常量池结束之后,接着就是访问标志(access)了,类型是U2也就是占两个字节。
注意,这个和前面的查找方式就有些不同了,不再是把16进制转化成10进制然后到表里面去找了。
而是表里面的每一项对应16个bit位中的一个,如果为真(1),就说明有这个限定符,否则就是没有(这么做真是省了不少空间啊)。
目前限定符只有8个,如下表所示,标志值代表所在的位置,其余的用补0。
我们找到文件中的标志位,值为“00 21”,转换为二进制就是“0000 0000 0010 0001”。
让我们再次打开Excel,表格图形走一走(这里笔者有个疑问,为什么不按顺序依次排列呢,有知道的朋友希望不吝赐教)。
我们就会看到,这个类的限定组合是ACC_PUBLIC+ACC_SUPER,6不6?
等等,看到这个“ACC_PUBLIC”,笔者不由得虎躯一震,不知道大家还记得前文提到过的ASM吗?
给大家上一小段写过的测试代码,这里肯定有一些不可告人的秘密!
六、继承与实现:其实就是结构表中的类索引(this_class),父类索引(super_class)和接口索引个数(interfaces_count)+接接口索引(interface)。
这里简单说一下这个“索引”,如果说数据库的索引,大家肯定都知道,是用来快速查找的。其实这里的意思也差不多,当所有的类都加载到内存中,会生成一个包名+类名的唯一标识,就是所说的这个“索引”,虚拟机就可以根据类索引找到这个类。
为什么接口和类的结构不一样?因为java是单根继承,只能继承一个类或接口;但是却可以实现多个接口。
为了直观一些,我们把执行javap反编译的截图再拿回来。
类索引(this_class):类型为u2,指向常量池中的某个常量。对应的值是“0x0005”,即指向#5->#26,也就是com/xkx/App。
父类索引(super_class):类型为u2,指向常量池中的某个常量。对应的值是“0x0006”,即指向#6->#27,也就是java/lang/Object。
接口索引个数(interfaces_count):类型为u2,记录了实现接口的个数。和常量池其实是一样的,先记录一个数量,后面是对应的数据,笔者把它理解为Map
interfaces_count,Object[] interfaces>。
接口索引(interface):接口的索引,如果上面的个数为0,那么这部分就没有。如果有的话,每一个接口索引占两个字节,也是指向一个接口的地址。
好了,今天就写这么多吧,明天继续!
喜欢文章或想一起学习的朋友可以关注我,给我点赞,我将会持续更新,有什么疑问或文中有不当之处请给我留言,真诚地希望能与大家一起交流探讨,学习进步。