java总结- 字符串类型内存模型(完善中)

参考地址:class常量池、字符串常量池和运行时常量池的区别_那年那些事儿-CSDN博客_运行时常量池和字符串常量池的区别

   public static void main(String[] args) {
        String s1="hello";
        String s2="hello";
        String s3=new String("world");
        String s4=s3.intern();
        String s5="world";
        String s6="helloworld";
        String s7=s1+s5;
        System.out.println(s1 == s2);//true
        System.out.println(s3 == s4);//false
        System.out.println(s4 == s5);//true
        System.out.println(s6 == s7);//false
    }

解析程序的内存分配过程:

1.首先,解析第一句的时候,在堆中会有hello的实例对象(驻留字符串对象),全局StringTable(在 HotSpot VM 里实现的 String Pool 功能的是一个 StringTable 类,它是一个哈希表)中存放"hello"的一个引用值。

2.与第一句是不同的实例,当在解析第二句的时候查找StringTable,里面有hello的全局驻留字符串的引用,所以s2的引用地址与s1的引用地址相同。所以是s1=s2.

3.然后解析到第三句,堆中会生成两个实例对象,一个是world的实例对象(驻留字符串对象),并且在StringTable中存储一个world的引用值,另外一个是new出来的world的实例对象,返回对象的引用。整个过程可以看做是以下步骤的:String tmp="world";String s3=new String(tmp);

4.解析到第四句,调用s3的intern函数,返回StringTable中world的引用值,如果StringTable中没有,就把s3的引用值添加进去,这里已经有了,所以返回tmp中world的引用值。所以s3!=s4。

5.解析到第五句,返回存在StringTable中world的引用值。所以s4=s5。

6.解析到第六句,过程同第1步,在堆中会有helloworld的主流字符串对象,并在StringTable中会存放它的引用。

7.解析第七句,在s1+s5的过程,jdk1.5之后编译器有个优化,会自动把"+"替换成StringBulider的append,先用StringBuilder stb=new StringBuilder(s1的驻留字符串对象)生成一个StringBuilder对象,stb.append(s5的驻留字符串对象),接着调用StringBuilder的toString()方法在堆中创建一个String对象,最后返回生成的String对象的地址引用。所以s6!=s7

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值