8. StringTable

一、String 的存储结构和StringPool

1.1 String存储结构变更

​ String在jdk9中底层不再使用char型数组,而是使用byte型数组,因为在实际开发中发现大多数字符串中存储的都是Lantin-1(每个字符一个字节),这些字符只需要1byte的空间就可以存储。

1.2 String的基本特性

  • 字符串常量池中不会存储相同内容的字符串;
  • String的String Pool是一个固定大小的Hashtable(里面只存储String的引用,并不存储实际内容,实际内容存储在运行时常量池),默认大小长度是60013。如果长度过短,就会因为String的数量增加从而导致链表变长,效率降低;
  • 使用-XX:StringTableSize可设置StringTable的长度,1009是可设置的最小长度。

二、String 的内存分配

​ 看 6. JVM 方法区的3.1

三、字符串拼接操作

  1. 常量与常量的拼接结果在常量池,原理是编译期优化;
  2. 常量池中不会存在相同内容的常量;
  3. 只要其中有一个是变量,结果就在堆中,变量拼接的原理是StringBuilder
  4. 如果拼接的结果调用intern()方法,则主动将常量池中还没有的字符串对象放入池中,并返回此对象的地址。
public class Test3 {
    public static void main(String[] args) {
        String a = "abcdef";
        String b = "abc";
        String c = "def";
        String d = "abc" + "def";   //"abc" + "def" 在编译期间就被优化成为了 "abcdef"
        String e = b + c;   //本质上执行的代码如下:String e = (new StringBuilder()).append(b).append(c).toString();

        System.out.println(d == a);     //true,因为d和a指向常量池中同一个地址
        System.out.println(e == a);     //false,因为e指向推空间中一个StringBuilder.toString()所在的地址,而a指向常量池中的一个地址
        System.out.println(e.intern() == a);       //true,e.intern()返回的是字符串"abcdef"在常量池中的地址
    }
}

注意:只有字符串变量在拼接的时候调用的才是StringBuilder,如果是final修饰的变量,则是编译期优化

public void test2() {
    final String a = "abcdef";
    final String b = "abc";
    final String c = "def";
    String d = b + c;   //b + c在编译期间就被优化成为了 "abcdef"

    System.out.println(d == a);     //true
}

四、intern() 的使用

4.1 intern() 的基本原理

​ 如果不是用双引号声明的String对象,可以使用String提供的intern方法:inter方法会从字符串常量池中查询当前字符串是否存在,若存在,返回已有的串池中对象的地址;若不存在,会把对象的引用地址复制一份,放入串池,并返回串池中的引用地址。

​ intern 就是确保字符串在内存里只有一份拷贝,这样可以节约空间,加快字符串操作任务的速度。

4.2 new String(“ab”) 会创建几个对象,new String(“a”) + new String(“b”) 呢

  1. new String(“ab”) 会创建2个对象

    • new String本身
    • 字符串常量池中的ab

在这里插入图片描述

  1. new String(“a”) + new String(“b”) 会创建5个对象

    • StringBuilder
    • new String(“a”)
    • 常量池中的a
    • new String(“b”)
    • 常量池中的b

在这里插入图片描述

4.2 关于 intern 的题目

练习一

public void test4() {
    String s = new String("1");     //在堆空间中创建了String对象,在常量池中创建了“1”
    String s1 = s.intern();
    String s2 = "1";

    System.out.println(s == s1);    //false,s指向堆空间中的String对象,s1指向常量池中的 “1”
    System.out.println(s1 == s2);   //true,s2也指向常量池中的“1”
}

练习二

public void test5() {
    String s3 = new String("1") + new String("1");      //此时字符串常量池中只有 “1”,而没有 “11”
    s3.intern();    //jdk7开始,常量池中并没有创建“11”,而是创建一个指向堆空间中 new String("11")的地址
    String s4 = "11";   //s4变量记录的地址:指向堆空间中 new String("11")的地址

    System.out.println(s3 == s4);   //jdk6:false,jdk7/8:true

    
}

在这里插入图片描述

在这里插入图片描述

练习三

public void test6() {
    String s5 = "11";
    String s6 = new String("1") + new String("1");
    String s7 = s6.intern();

    System.out.println(s5 == s6);   //false,因为“11”已经先创建了,所以调用s6.intern后,并不会在常量池中创建指向s5地址的项
    System.out.println(s5 == s7);   //true
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值