4000336032

成都Java培训学校

Java常量池与字符串intern——汇道科技

来源:成都Java培训学校 发布时间:2016/10/17 10:30:53

作为目前计算机编程语言榜名的语言,Java在计算机科学知识时代扮演了重要的角色。有超过500万Java开发者活跃在世界的每一个角落,数以亿万计的Web用户每次上网都亲历Java的威力和感受其魅力,吸引着越来越多的互联网爱好者加入Java培训队伍中。

汇道科技坚持“成就学生的心一百年不变”的理念,打造出较适合中国企业需求的Java人才培养体系。Java课程是汇道科技的大课程方向,为学员打造较人性化的Java课程。下面就跟着汇道科技一起来认识Java常量池与字符串intern。


   在Java应用程序运行时,Java虚拟机会保存一份内部的运行时常量池,它区别于class文件的常量池,是class文件常量池映射到虚拟机中的数据结构。


1.CONSTANT_Class入口解析

数组类的符号解析较为特殊。若是基本类型数组,那么虚拟机将创建该基本类型的新数组类,并创建一个Class实例来代表该类型,数组类的定义类加载器为 启动类加载器。若是引用类型的数组,那么在此之前还会进行引用类型的解析,数组类的定义类加载器为引用类型的定义类加载器。

非数组类和接口的的解析将经历以下步骤:

(1) 加载该类型和其所有的超类型

如果该类型在此之前已经装载到了虚拟机的当前命名空间,那么直接使用已经被装载的类型即可,否则由引用的发起类的初始类加载器进行加载。对目标类型的超类的加载必然是在对当前类型加载完的基础上进行的,因为只有加载完当前类型,才能从class文件的super_class域找到其直接超类的符号引用,再递归进行解析和加载,直至java.lang.Object类。而在递归返回的过程中,会检查interfaces域以查看实现或扩展了哪些接口,并再次递归遍历对接口的符号引用。

(2)检查访问权限

随后是对目标类型的连接和初始化,这样才可以正常使用该类型。前面提到,对目标类型的初始化需要其所有超类都必须进行初始化(超接口不是必须的),并且,由于已经对其超类进行了加载,所以不必再依赖于自该类向Object类的解析顺序,而是从Object类向该类进行初始化。类型的连接和初始化步骤如下:

(3)类型校验

(4)类型准备

(5)类型解析(可推迟)

注意该过程是对被引用类型及其超类的符号引用的解析,因为对于被引用类型的某些符号引用不会立刻用到,故该步骤之前是严格意义上属于发起引用的类型的符号 解析的过程。只有在主动使用被引用类型的这些符号引用所指向的类型时,才会对这些符号引用进行解析,对其所指向的类型进行装载、连接和初始化。

(6)类型初始化


2.CONSTANT_Fieldref入口解析

由于一个类型不会含有其超类型所定义的字段,所以对目标字段的搜索将会从字段所 指向的类型开始,从该类型开始搜索,再递归搜索其所实现或扩展的接口,再递归搜索其超类,直至找到目标字段,并会将运行时常量池的该字段入口标记为已解 析,并在该常量池的数据上改为对这个字段的直接引用。

3.CONSTANT_Methodref入口解析

与字段的搜索类似但有所不同,其搜索顺序将从该类型开始,再递归搜索其超类,在递归搜索其所实现或扩展的接口。


4.CONSTANT_InterfaceMethodRef入口解析

对接口方法的搜索就是从被解析的接口开始,向其超接口递归搜索。


5.CONSTANT_String入口解析

Java虚拟机会将字符串处理为一个字符串对象加以维护,而虚拟机所维护的就是一张 字符串池,它包含所有被”拘留”的字符串对象的引用。对CONSTANT_String常量池的解析首先就要查看字符串池中该字符串对象的引用是否存在, 如果存在则直接把常量池数据解析为该字符串对象的引用,若不存在,那么就需要根据这个字符串序列创建一个字符串对象,并将其引用加入到字符串池中,并将常 量池数据解析为该引用。

也可以使用String对象的intern对象来拘留一个字符串(注意并非字符串对象),若该字符串池中存在对该字 符串序列的对象的引用,那么直接返回该引用即可,否则,将会拘留该字符串,但注意拘留返回的字符串对象引用将不会指向原String对象,因为原 String对象位于Java堆,而字符串池的对象是虚拟机所创建的,由虚拟机所维护。

package com.ice.intern;

public class InternTest {

public static void main(String args[]){

String a = new String("123");

String b = a;

String c = new String("123");;

System.out.println("before intern:");

System.out.println("a = b ? :" + (a == b));

System.out.println("a = c ? :" + (a == c));

a = a.intern();

c = c.intern();

System.out.println("after intern:");

System.out.println("a = b ? :" + (a == b));

System.out.println("a = c ? :" + (a == c));

}

}

结果如下:



(6).其他类型(数据基本类型)入口解析

直接使用常量池所包含的常量值即可


6.直接引用

常量池解析较终将符号引用替换成为直接引用。指向类型、类变量和类方法的直接引用可能为在方法区的指针。而指向实例变量和实例方法的直接引用是从对象映像的开始到该实例变量或方法表的偏移。

实例变量的组织方式为:从Object类开始到该实例的类型,将类中声明的实例变量按在class文件中出现的顺序依次放在对象映像中。

实例方法的组织方式较为类似:从Object类开始到该实例的类型,将类中声明的实例方法指针按在class文件中出现的顺序依次放在对象映像中。但对于重写的方法将出现在超类对应的位置(该方法次出现的位置)。

但是访问接口方法就不能简单地通过方法表的偏移量来进行访问,而必须搜索对象的类的方法表来找到该方法。

比如Factory接口分别由A和B来实现其produce()方法,但由于A和B不能增加由同一个实现了Factory接口的超类派生,即有着同样的produce()方法偏移,那么就无法通过方法表的偏移来访问Factory的produce()方法。


7.装载约束

对于一个类型指向另一个类型的符号引用,如果引用的类型和被引用类型并非由同一个初始加载器加载(可能通过用户自定 义ClassLoader来实现),那么虚拟机就必须确保被引用类型在不同的命名空间中保持一致。这样就通过自定义ClassLoader来加载不受信类 型后,就不会发生解析对被引用类型的符号引用时,把受信的类型当做已经被解析过的不受信类型(因为对方法的符号引用只有权限定名和描述符,并不会也无法得知其初始类加载器),从而调用了不受信类型的方法访问受信类型的受保护成员。



PS:

? 每周汇道公开课,导师为您答疑,实际案例实际分析

? 行业专场风暴:JAVA技术,H5前端,UI设计,Android等IT技术专场

? 线下聚会交流:与腾讯、华为、阿里、百度等分享交流

? 资源及人脉共享

? 名企就业岗位推荐

? 其他不定期活动


汇道IT社群是基于互联网IT技术的学习型组织,社群设有每周风暴,嘉宾分享,问题答疑等活动。另外,社群已开设行业交流专场,如JAVA技术,H5前端,UI设计,Android等IT专场。欢迎各行业互联网IT技术达人加入我们汇道大家庭,IT都在这,赶紧行动吧!

更多优惠信息请咨询成都Java培训学校http://cdjavapx.soxsok.com/

联系人:孙老师

联系电话:400-0088-291  在线qq:1974950309



领取试听课
每天限量名额,先到先得
温馨提示:为不影响您的学业,来校区前请先电话或QQ咨询,方便我校安排相关的专业老师为您解答
  • 详情请进入 成都Java培训学校

关于我们 | 招生信息 | 新闻中心 | 学校动态

版权所有:搜学搜课(www.soxsok.com)