《剑指JVM》——第12章——不同方式的字符串拼接的地址共享问题——字符串常量池2

🌈hello,你好鸭,我是Ethan,西安电子科技大学大三在读,很高兴你能来阅读。

✔️目前博客主要更新Java系列、项目案例、计算机必学四件套等。
🏃人生之义,在于追求,不在成败,勤通大道。加油呀!

🔥个人主页:Ethan Yankang
🔥推荐:史上最强八股文||一分钟看完我的几百篇博客

🔥温馨提示:划到文末发现专栏彩蛋   点击这里直接传送

🔥本篇概览:详细讲解了不同方式的字符串拼接的地址共享问题🌈⭕🔥


【计算机领域一切迷惑的源头都是基本概念的模糊,算法除外】


🔥 《剑指JVM》序言-CSDN博客

🔥 《剑指JVM》全书-CSDN博客


🌈章节引出

前一篇章:《剑指JVM》——第12章——String的基本特性——字符串常量池1-CSDN博客

🌈章节速览


在日常开发中,大家会经常用到字符串的拼接,字符串的拼接通常使用“+”或 String 类的concatO)方法,它们有什么不同呢?另外,使用针对字符串常量拼接和字符串变量拼接又有什么区别呢?字符串拼接结果存放在常量池还是堆中呢?通过运行和分析下面的代码,相信你可以得出结论


12.3.1 不同方式的字符串拼接的地址共享问题

0.开发必备DEBUG显示堆中内存值:

【验证不同字符串的拼接方式的存放地址的差异】

前提条件:

虽然在JavaJDK8之后的版本中,将字符串常量池放入了堆中但是在堆区域中的new出来的对象与字符串常量池中的对象是隔开的。


代码如下:

1.字面量直接相加

 @Test
    public void test1() {
        String s1 = "a" + "b" + "c"; // 编译期优化:等同于 "abc"
        String s2 = "abc"; // "abc"一定是放在 StringTable 中
        System.out.println(s1 == s2); // true
    }

由于编译器优化,"a"+"b"+"c"="abc",在编译期就已经决定好了。

                        


2.+拼接中出现字符串变量或者直接new的

底层都是在堆中重新创建对象。结果都不放在字符串常量池中,地址自然不相同。都是new出来的,堆中的地址。。

 @Test
    public void test2() {
        String s1 = "javaEE";
        String s2 = "hadoop";
        String s3 = "javaEEhadoop";
        String s4 = "javaEE" + new String("hadoop");
        String s5 = s1 + "hadoop";
        String s6 = "javaEE" + s2;
        String s7 = s1 + s2;
        // "+”拼接中出现字符串变量等非字面常量
        // 结果都不在 stringTable
        System.out.println(s3 == s4); // false
        System.out.println(s3 == s5); // false
        System.out.println(s3 == s6); // false
        System.out.println(s3 == s7); // false
        System.out.println(s5 == s6); // false
        System.out.println(s5 == s7); // false
        System.out.println(s6 == s7); // false
    }

运行细节如下:


【(实际全部都不一样,我就不一一展示了)】


3.使用concat函数拼接的

底层还是new的对象,不共享地址。

 @Test
    public void test3() {
        String s1 = "javaEE";
        String s2 = "hadoop";
        String s3 = "javaEEhadoop";
        String s4 = s1.concat(s2);
        // concat 拼接结果不在 StringTable
        System.out.println(s3 == s4); // false
    }


4.使用intern()将堆中对象转移至常量池

这样会与堆中原有的引用共享对象。

@Test
    public void test4() {
        String s1 = "hello";
        String s2 = "java";
        String s3 = "hellojava";
        String s4 = (s1 + s2).intern();
        String s5 = s1.concat(s2).intern();
        // 拼接后调用 intern()方法,结果都在 stringTable 中
        System.out.println(s3 == s4);
        System.out.println(s3 == s5);
    }


5.final修饰的变量仍可以等价于字面量

 @Test
    public void test5() {
        final String s1 = "hello";
        final String s2 = "java";
        String s3 = "hellojava";
        String s4 = s1 + s2;
        System.out.println(s3 == s4); // true
    }

总结字符串拼接的地址共享问题 



$$自己逐步分析

package com.edu;

import org.junit.jupiter.api.Test;

public class StringTable {
    @Test
    void test1(){
        String s1="hello";
        String s2="hello";
        String s3="zyk";
        String s4=new String("xidian");
        String s5=new String("xidian");
        String s6="xidian";
        String s7="xi"+new String("dian");
        System.out.println(s6==s7);
        String s8=s7.intern();
        System.out.println(s6==s8);
    }
}

执行s1之前

s1执行了后

s2执行后

【由此看出,字面量直接是共享字符串地址的】

s4执行后

s5执行后

【由此看出,new出来的两个变量是直接在堆中创建的新对象,不共享地址】

【由此看出,拼接出来的变量是直接在堆中创建的新对象,不共享地址】

【由此看出,intern()操作是将字符串放入常量池中,会与之前的s4共享对象地址】

注意,哈希冲突:

 【上面的s5和s6是不同的地址,却有着相同的哈希值,这就是哈希冲突。(一般来说不会发生。)】



💖💖💖💖💖💖💖💖💖💖💖💖💖💖💖💖💖💖

热门专栏推荐

🌈🌈计算机科学入门系列                     关注走一波💕💕

🌈🌈CSAPP深入理解计算机原理        关注走一波💕💕

🌈🌈微服务项目之黑马头条                 关注走一波💕💕

🌈🌈redis深度项目之黑马点评            关注走一波💕💕

🌈🌈JAVA面试八股文系列专栏           关注走一波💕💕

🌈🌈JAVA基础试题集精讲                  关注走一波💕💕   

🌈🌈代码随想录精讲200题                  关注走一波💕💕


总栏

🌈🌈JAVA基础要夯牢                         关注走一波💕💕  

🌈🌈​​​​​​JAVA后端技术栈                          关注走一波💕💕  

🌈🌈JAVA面试八股文​​​​​​                          关注走一波💕💕  

🌈🌈JAVA项目(含源码深度剖析)    关注走一波💕💕  

🌈🌈计算机四件套                               关注走一波💕💕  

🌈🌈数据结构与算法                           ​关注走一波💕💕  

🌈🌈必知必会工具集                           关注走一波💕💕

🌈🌈书籍网课笔记汇总                       关注走一波💕💕         



📣非常感谢你阅读到这里,如果这篇文章对你有帮助,希望能留下你的点赞👍 关注❤收藏✅ 评论💬,大佬三连必回哦!thanks!!!
📚愿大家都能学有所得,功不唐捐!

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值