String类相关(底层,创建过程等)解读

String类详解

源码:
public final class String
    implements java.io.Serializable, Comparable<String>, CharSequence {
    /** The value is used for character storage. */
    private final char value[];

    /** Cache the hash code for the string */
    private int hash; // Default to 0
    //...
}

String底层是被final关键字修饰的char数组,String类不能被继承,它的成员方法都默认为final方法,字符串一旦创建就不能再修改。

内存分配:

在Java的内存分配中,总共3种常量池,分别是Class常量池运行时常量池字符串常量池

JVM为了提高性能和减少内存的开销,在实例化字符串的时候进行了一些优化:使用字符串常量池。每当创建字符串常量时,JVM会首先检查字符串常量池,如果该字符串已经存在常量池中,那么就直接返回常量池中的实例引用。如果字符串不存在常量池中,就会实例化该字符串并且将其放到常量池中。由于String字符串的不可变性,常量池中一定不存在两个相同的字符串

对象生成方案:
  • 通过字面量赋值创建字符串

    如 String s = “test”, 可能创建一个或者不创建对象。先在常量池中查找是否存在相同的字符串(如 String s = “test”),若存在,则将栈中的引用直接指向该字符串;若不存在,则在常量池中生成一个字符串,再将栈中的引用指向该字符串。
    在这里插入图片描述

  • 通过new关键字创建字符串

    如 String s = new String(“test”), 至少会创建一个对象,也可能会创建两个对象。因为用到new关键字,肯定会在堆中创建一个String对象,如果字符池中已经存在"test",则不会在字符串池中创建一个String对象,如果不存在,则会在字符串常量池中也创建一个对象。
    在这里插入图片描述

intern方法:

返回字符串对象的规范化表示形式

  • 1.7之前

    查询常量池中是否有字符串存在,如果存在,则返回常量池中的引用。如果在常量池找不到对应的字符串,则将字符串拷贝到常量池中。

  • 1.7之后

    查询常量池中是否有字符串存在,如果存在,则返回常量池中的引用。如果在常量池找不到对应的字符串,则不会再将字符串拷贝到常量池,而只是在常量池中生成一个对原字符串的引用。
    在这里插入图片描述

常量字符串的拼接操作:
  • 常量字符串的“+”操作,编译阶段直接会合成为一个字符串。如string str=”JA”+”VA”,在编译阶段会直接合并成语句String str=”JAVA”,于是会去常量池中查找是否存在”JAVA”,从而进行创建或引用。
  • 对于final字段,编译期直接进行了常量替换(而对于非final字段则是在运行期进行赋值处理的)。
    final String str1=”ja”;
    final String str2=”va”;
    String str3=str1+str2;
    在编译时,直接替换成了String str3=”ja”+”va”,根据第三条规则,再次替换成String str3=”JAVA”
  • 常量字符串和变量拼接时(如:String str3=baseStr + “01”;)会调用stringBuilder.append()在堆上创建新的对象。
示例在这里插入图片描述
String s = new String("test"System.out.println(s == s.intern())  //false

String s9 = new String("1") + new String("1");
s9.intern();
String s10 = "11";
System.out.println(s9 == s9.intern()); //true
System.out.println(s9 == s10);  //true

String ss1 = new String("nam");
String ss2 = new String("e");
String ss3 = ss1 + ss2;
String ss4 = ss3.intern();
String ss5 = "name";
System.out.println(ss3 == ss4);  //false
System.out.println(ss4 == ss5);  //true

其他相关

  • jdk1.7之前,常量池放在方法区,jdk1.7之后,常量池放在堆内存。
  • StringBuilder的toString()方法会创建一个新对象,底层为new String()

参考博文

几张图轻松理解String.intern()

String直接赋值与使用new String的区别

字符串常量池、运行时常量池、class常量池分别到底都在哪?

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值