《Java虚拟机原理图解》 1.2.2、Class文件里的常量池具体解释(上)
[last updated:2014/11/27]
NO1.常量池在class文件的什么位置?
我的上一篇文章《Java虚拟机原理图解》 1、class文件基本组织结构中已经提到了class的文件结构,在class文件里的魔数、副版本号号、主版本号之后,紧接着就是常量池的数据区域了,例如以下图用红线包含的位置:
知道了常量池的位置后,然后让我们来揭秘常量池里到底有什么东西吧~
NO2.常量池的里面是怎么组织的?
常量池的组织非常easy,前端的两个字节占有的位置叫做常量池计数器(constant_pool_count),它记录着常量池的组成元素 常量池项(cp_info)
的个数。紧接着会排列着constant_pool_count-1个常量池项(cp_info)。例如以下图所看到的:
NO3.常量池项 (cp_info) 的结构是什么?
每一个常量池项(cp_info)
都会相应记录着class文件里的某中类型的字面量。让我们先来了解一下常量池项(cp_info)的结构吧:JVM虚拟机规定了不同的tag值和不同类型的字面量相应关系例如以下:
所以依据cp_info中的tag 不同的值,能够将cp_info 更细化为下面结构体:
CONSTANT_Utf8_info,CONSTANT_Integer_info,CONSTANT_Float_info,CONSTANT_Long_info,
CONSTANT_Double_info,CONSTANT_Class_info,CONSTANT_String_info,CONSTANT_Fieldref_info,
CONSTANT_Methodref_info,CONSTANT_InterfaceMethodref_info,CONSTANT_NameAndType_info,CONSTANT_MethodHandle_info,
CONSTANT_MethodType_info,CONSTANT_InvokeDynamic_info。
如今让我们看一下细化了的常量池的结构会是类似下图所看到的的样子:
NO4.常量池可以表示那些信息?
NO5. int和float数据类型的常量在常量池中是如何表示和存储的?(CONSTANT_Integer_info,
CONSTANT_Float_info)
Java语言规范规定了 int类型和Float 类型的数据类型占用 4 个字节的空间。那么存在于class字节码文件里的该类型的常量是怎样存储的呢?对应地,在常量池中,将
int和Float类型的常量分别使用CONSTANT_Integer_info和 Constant_float_info表示,他们的结构例如以下所看到的:举例:建以下的类 IntAndFloatTest.java,在这个类中,我们声明了五个变量,可是取值就两种int类型的10 和Float类型的11f。
package com.louis.jvm; public class IntAndFloatTest { private final int a = 10;
private final int b = 10;
private float c = 11f;
private float d = 11f;
private float e = 11f; }然后用编译器编译成IntAndFloatTest.class字节码文件,我们通过javap
-v IntAndFloatTest 指令来看一下其常量池中的信息,能够看到尽管我们在代码中写了两次10 和三次11f,可是常量池中,就仅仅有一个常量10 和一个常量11f,例如以下图所看到的:从结果上能够看到常量池第#8 个常量池项(cp_info) 就是CONSTANT_Integer_info,值为10;第#23个常量池项(cp_info)
就是CONSTANT_Float_info,值为11f。(常量池中其它的东西先别纠结啦,我们会面会一一解说的哦)。代码中全部用到
int 类型 10 的地方,会使用指向常量池的指针值#8 定位到第#8 个常量池项(cp_info),即值为
10的结构体 CONSTANT_Integer_info,而用到float类型的11f时,也会指向常量池的指针值#23来定位到第#23个常量池项(cp_info)
即值为11f的结构体CONSTANT_Float_info。例如以下图所看到的:
NO6. long和 double数据类型的常量在常量池中是如何表示和存储的?(CONSTANT_Long_info、CONSTANT_Double_info
)
Java语言规范规定了
long 类型和 double类型的数据类型占用8 个字节的空间。那么存在于class 字节码文件里的该类型的常量是怎样存储的呢?对应地,在常量池中,将long和double类型的常量分别使用CONSTANT_Long_info和Constant_Double_info表示,他们的结构例如以下所看到的:举例:建以下的类 LongAndDoubleTest.java,在这个类中,我们声明了六个变量,可是取值就两种Long 类型的-6076574518398440533L 和Double 类型的10.1234567890D。
package com.louis.jvm; public class LongAndDoubleTest { private long a = -6076574518398440533L;
private long b = -6076574518398440533L;
private long c = -6076574518398440533L;
private double d = 10.1234567890D;
private double e = 10.1234567890D;
private double f = 10.1234567890D;
}然后用编译器编译成
LongAndDoubleTest.class 字节码文件,我们通过javap -v LongAndDoubleTest指令来看一下其常量池中的信息,能够看到尽管我们在代码中写了三次-6076574518398440533L 和三次10.1234567890D,可是常量池中,就仅仅有一个常量-6076574518398440533L 和一个常量10.1234567890D,例如以下图所看到的:从结果上能够看到常量池第
#18 个常量池项(cp_info) 就是CONSTANT_Long_info,值为-6076574518398440533L ;第 #26个常量池项(cp_info)
就是CONSTANT_Double_info,值为10.1234567890D。(常量池中其它的东西先别纠结啦,我们会面会一一解说的哦)。代码中全部用到 long 类型-6076574518398440533L 的地方,会使用指向常量池的指针值#18 定位到第 #18 个常量池项(cp_info),即值为-6076574518398440533L 的结构体CONSTANT_Long_info,而用到double类型的10.1234567890D时,也会指向常量池的指针值#26
来定位到第 #26 个常量池项(cp_info) 即值为10.1234567890D的结构体CONSTANT_Double_info。例如以下图所看到的:
NO7. String类型的字符串常量在常量池中是如何表示和存储的?(CONSTANT_String_info、CONSTANT_Utf8_info)
对于字符串而言,JVM会将字符串类型的字面量以UTF-8 编码格式存储到在class字节码文件里。这么说可能有点摸不着北,我们先从直观的Java源代码中中出现的用双引號"" 括起来的字符串来看,在编译器编译的时候,都会将这些字符串转换成CONSTANT_String_info结构体,然后放置于常量池中。其结构例如以下所看到的:
如上图所看到的的结构体,CONSTANT_String_info结构体中的string_index的值指向了CONSTANT_Utf8_info结构体,而字符串的utf-8编码数据就在这个结构体之中。例如以下图所看到的:
请看一例,定义一个简单的StringTest.java类,然后在这个类里加一个"JVM原理" 字符串,然后,我们来看看它在class文件里是如何组织的。
package com.louis.jvm; public class StringTest {
private String s1 = "JVM原理";
private String s2 = "JVM原理";
private String s3 = "JVM原理";
private String s4 = "JVM原理";
}将Java源代码编译成StringTest.class文件后,在此文件的文件夹下运行 javap -v StringTest 命令,会看到例如以下的常量池信息的轮廓:
(PS :使用javap -v 指令能看到易于我们阅读的信息,查看真正的字节码文件能够使用HEXWin、NOTEPAD++、UtraEdit 等工具。)
在面的图中,我们能够看到CONSTANT_String_info结构体位于常量池的第#15个索引位置。而存放"Java虚拟机原理"
字符串的 UTF-8编码格式的字节数组被放到CONSTANT_Utf8_info结构体中,该结构体位于常量池的第#16个索引位置。上面的图仅仅是看了个轮廓,让我们再深入地看一下它们的组织吧。请看下图:由上图可见:“JVM原理”的UTF-8编码的数组是:4A564D
E5 8E 9FE7 90 86,而且存入了CONSTANT_Utf8_info结构体中。
NO8. 类文件里定义的类名和类中使用到的类在常量池中是如何被组织和存储的?(CONSTANT_Class_info)
JVM会将某个Java 类中全部使用到了的类的全然限定名 以二进制形式的全然限定名 封装成CONSTANT_Class_info结构体中,然后将其放置到常量池里。CONSTANT_Class_info
的tag值为 7 。其结构例如以下:Tips:类的全然限定名和二进制形式的全然限定名
在某个Java源代码中,我们会使用非常多个类,比方我们定义了一个 ClassTest的类,并把它放到com.louis.jvm
包下,则 ClassTest类的全然限定名为com.louis.jvm.ClassTest,将JVM编译器将类编译成class文件后,此全然限定名在class文件里,是以二进制形式的全然限定名存储的,即它会把全然限定符的"."换成"/"
,即在class文件里存储的 ClassTest类的全然限定名称是"com/louis/jvm/ClassTest"。由于这样的形式的全然限定名是放在了class二进制形式的字节码文件里,所以就称之为 二进制形式的全然限定名。举例,我们定义一个非常easy的ClassTest类,来看一下常量池是怎么对类的全然限定名进行存储的。
package com.jvm;
import java.util.Date;
public class ClassTest {
private Date date =new Date();
}将Java源代码编译成ClassTest.class文件后,在此文件的文件夹下运行
javap -v ClassTest 命令,会看到例如以下的常量池信息的轮廓:如上图所看到的,在ClassTest.class文件的常量池中,共同拥有 3 个CONSTANT_Class_info结构体,分别表示ClassTest
中用到的Class信息。 我们就看当中一个表示com/jvm.ClassTest的CONSTANT_Class_info
结构体。它在常量池中的位置是#1,它的name_index值为#2,它指向了常量池的第2 个常量池项,例如以下所看到的:注意:
对于某个类而言,其class文件里至少要有两个CONSTANT_Class_info常量池项,用来表示自己的类信息和其父类信息。(除了java.lang.Object类除外,其它的不论什么类都会默认继承自java.lang.Object)假设类声明实现了某些接口,那么接口的信息也会生成相应的CONSTANT_Class_info常量池项。
除此之外,假设在类中使用到了其它的类,仅仅有真正使用到了对应的类,JDK编译器才会将类的信息组成CONSTANT_Class_info常量池项放置到常量池中。例如以下图:
package com.louis.jvm; import java.util.Date; public class Other{
private Date date; public Other()
{
Date da;
}
}上述的Other的类,在JDK将其编译成class文件时,常量池中并没有java.util.Date相应的CONSTANT_Class_info常量池项,为什么呢?
在Other类中尽管定义了Date类型的两个变量date、da,可是JDK编译的时候,觉得你仅仅是声明了“Ljava/util/Date”类型的变量,并没有实际使用到Ljava/util/Date类。将类信息放置到常量池中的目的,是为了在兴许的代码中有可能会重复用到它。非常显然,JDK在编译Other类的时候,会解析到Date类有没实用到,发现该类在代码中就没实用到过,所以就觉得没有必要将它的信息放置到常量池中了。
将上述的Other类改写一下,仅使用new Date(),例如以下图所看到的:
package com.louis.jvm; import java.util.Date; public class Other{
public Other()
{
new Date();
}
}这时候使用javap -v Other ,能够查看到常量池中有表示java/util/Date的常量池项:
总结:
1.对于某个类或接口而言,其自身、父类和继承或实现的接口的信息会被直接组装成CONSTANT_Class_info常量池项放置到常量池中;
2. 类中或接口中使用到了其它的类,仅仅有在类中实际使用到了该类时,该类的信息才会在常量池中有相应的CONSTANT_Class_info常量池项;
3. 类中或接口中仅仅定义某种类型的变量,JDK仅仅会将变量的类型描写叙述信息以UTF-8字符串组成CONSTANT_Utf8_info常量池项放置到常量池中,上面在类中的private Date date;JDK编译器仅仅会将表示date的数据类型的“Ljava/util/Date”字符串放置到常量池中。
作者的话
本文是《Java虚拟机原理图解》系列的当中一篇,假设您有兴趣,请关注该系列的其它文章~
认为本文不错,顺手点个赞哦~~您的鼓舞,是我继续分享知识的强大动力!
-----------------------------------------------------------------------------------------------------------------------------------------
本文源自 http://blog.csdn.net/luanlouis/,如需转载,请注明出处,谢谢!
《Java虚拟机原理图解》 1.2.2、Class文件里的常量池具体解释(上)的更多相关文章
- Java虚拟机原理图解-- 1.2、class文件中的常量池
了解JVM虚拟机原理 是每一个Java程序员修炼的必经之路.但是由于JVM虚拟机中有很多的东西讲述的比较宽泛,在当前接触到的关于JVM虚拟机原理的教程或者博客中,绝大部分都是充斥的文字性的描述,很难给 ...
- 《Java虚拟机原理图解》1.4 class文件里的字段表集合--field字段在class文件里是如何组织的
0.前言 了解JVM虚拟机原理是每个Java程序猿修炼的必经之路.可是因为JVM虚拟机中有非常多的东西讲述的比較宽泛.在当前接触到的关于JVM虚拟机原理的教程或者博客中,绝大部分都是充斥的文字性的描写 ...
- 《Java虚拟机原理图解》1.4 class文件中的字段表集合--field字段在class文件中是怎样组织的
0.前言 了解JVM虚拟机原理是每一个Java程序员修炼的必经之路.但是由于JVM虚拟机中有很多的东西讲述的比较宽泛,在当前接触到的关于JVM虚拟机原理的教程或者博客中,绝大部分都是充斥的文字性的描述 ...
- 《Java虚拟机原理图解》 1.2.3、Class文件里的常量池具体解释(下)
NO9.类中引用到的field字段在常量池中是如何描写叙述的?(CONSTANT_Fieldref_info, CONSTANT_Name_Type_info) 一般而言.我们在定义类的过程中会定义一 ...
- Java虚拟机原理图解-- 1.1、class文件基本组织结构 [转]
作为Java程序猿,我们知道,我们写好的.java 源代码,最后会被Java编译器编译成后缀为.class的文件,该类型的文件是由字节组成的文件,又叫字节码文件.那么,class字节码文件里面到底是有 ...
- 《Java虚拟机原理图解》 1.2、class文件里的常量池
[最新更新:2014/11/11] 了解JVM虚拟机原理 是每个Java程序猿修炼的必经之路. 可是因为JVM虚拟机中有非常多的东西讲述的比較宽泛.在当前接触到的关于JVM虚拟机原理的教程或者博客中 ...
- 《Java虚拟机原理图解》1.5、 class文件中的方法表集合--method方法在class文件中是怎样组织的
0. 前言 了解JVM虚拟机原理是每一个Java程序员修炼的必经之路.但是由于JVM虚拟机中有很多的东西讲述的比较宽泛,在当前接触到的关于JVM虚拟机原理的教程或者博客中,绝大部分都是充斥的文字性的描 ...
- 《Java虚拟机原理图解》 1.2.2、Class文件中的常量池详解(上)
我的上一篇文章<Java虚拟机原理图解> 1.class文件基本组织结构中已经提到了class的文件结构,在class文件中的魔数.副版本号.主版本之后,紧接着就是常量池的数据区域了,如下 ...
- 《Java虚拟机原理图解》 1.2、class文件中的常量池
了解JVM虚拟机原理 是每一个Java程序员修炼的必经之路.但是由于JVM虚拟机中有很多的东西讲述的比较宽泛,在当前接触到的关于JVM虚拟机原理的教程或者博客中,绝大部分都是充斥的文字性的描述,很难给 ...
随机推荐
- Windows上部署Redis
http://www.cnblogs.com/gaobing/p/5026136.html
- 拦截API 注入进程
本文详细的介绍了在Visual Studio(以下简称VS)下实现API钩子的编程方法,阅读本文需要基础:有操作系统的基本知识(进程管理,内存管理),会在VS下编写和调试Win32应用程序和动态链接库 ...
- 自增或自减例子:i++和++i的相同点和不同点
/* Name:++i和i++的区别 Copyright: By.不懂网络 Author: Yangbin Date:2014年2月15日 02:40:27 Description:熟悉前自增或自减的 ...
- 转: Promises与Javascript异步编程
在如今都追求用户体验的时代,Ajax应用真的是无所不在.加上这些年浏览器技术.HTML5以及CSS3等的发展,越来越多的富Web应用出现:在给与我们良好体验的同时,Web开发人员在背后需要处理越来越多 ...
- MockObject
http://www.mockobjects.com/ http://jmock.org/download.html https://jakarta.apache.org/cactus/mock_vs ...
- Qt双缓冲机制:实现一个简单的绘图工具(纯代码实现)
http://blog.csdn.net/rl529014/article/details/51658350
- 基于纯Java代码的Spring容器和Web容器零配置的思考和实现(3) - 使用配置
经过<基于纯Java代码的Spring容器和Web容器零配置的思考和实现(1) - 数据源与事务管理>和<基于纯Java代码的Spring容器和Web容器零配置的思考和实现(2) - ...
- zookeeper 伪集群模式
问题二:开发没有足够机器,一台机子上是否装三个zookeeper服务器集群. 问题解答: 这种安装模式只能说是一种伪集群模式.三个zookeeper服务器都安装在同一个服务器(platform)上,需 ...
- Jquery读取URL参数
$(document).ready(function(){ function getQueryString(name) { var reg = new RegExp("(^|&)&q ...
- java类的结构(属性、方法、构造函数)
一.类的定义形式类定义的一般形式如下 [类定义修饰符] class <类名> { //类体 [成员变量声明] [构造函数] [成员方法] } 前面说过,在描述java语法时,方括号中 ...