intern方法详解,看完彻底理解字符串对象创建

本文详细介绍了Java中的字符串常量池和intern方法。从JDK6到JDK7的变化,字符串常量池的位置从永久代转移到堆中。讨论了四种创建字符串对象的方式:直接声明、使用new、+号拼接以及对象的toString方法,并解释了它们在常量池中的表现。此外,解释了intern方法的工作原理,当常量池中已存在相等字符串时返回其引用,否则将当前字符串添加到常量池。最后,文章通过实例总结了字符串创建的重要知识点。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1、字符串常量池与intern方法

字符串常量池

jdk6中字符串常量池在永久代,从jdk7开始放到了堆空间中(是堆中又划分了一块区域,注意这个细节!)。

字符串对象创建方式

1、直接使用双引号声明出来的String对象

会去字符串常量池中找(通过equals)是否已存在该对象或者它的引用(保存了引用的情况),不存在直接创建在常量池中,存在就指向存在的。

String s1 = "abc";//去字符串常量池中找“abc”,没有找到然后在字符串常量池创建了一个字符串对象
String s2 = "abc";//找到了直接指向字符串常量池的对象
s1 == s2 //true

2、通过new String()创建

会在堆的字符串常量池和常量池之外各创建一个对象

3、+号拼接

+号拼接两边只要有一个变量最后拼接出来的字符串对象都是只在堆中常量池之外,常量池中没有。

注意:拼接的怎样才算变量和常量?对象或基本类型变量都算变量,final修饰基本类型就为常量,修饰引用变量该引用变量拼接还是算变量拼接

//两个变量一共创建了6个对象,多的一个就是StringBuilder。(常量池没有)
String s = new String("ab") + new String("cd");
//一个变量都没有:两个都是常量(字符串本身是常量),此时创建的对象在字符串常量池中
final int a = 1;
String s2 = a + "cd"
String s = "ab" + "cd";

4、对象的toString方法

这个得看toString的具体实现,是用的new String()还是字符串与变量拼接。

intern方法

intern方法的作用是如果字符串常量池已经包含一个等于(通过equals方法比较)此String对象的字符串,则返回字符串常量池中这个字符串的引用, 否则将当前String对象的引用地址(堆中对象的引用地址)添加(或者叫复制)到字符串常量池中并返回

(JDK7之前字符串常量池不在堆中,因此那时候不是存引用,而是新创建一个相同的对象在方法区的字符串常量池中)

代码讲解

加号拼接、引号创建两者与intern混合使用

int n = 2;//变量
String s1 = n + "b";//只要有一个变量最后拼接出来的字符串对象都是只在堆中常量池之外,常量池中没有。
String s2 = s1.intern();//intern检查到常量池没有该字符串于是就将该字符串的地址引用保存到常量池中并返回
System.out.println(s1 == s2);//true,两个引用变量实际指向的都是在堆中字符串常量池外的实际字符串变量地址
String s = "2b";//通过引号创建会先去常量池找是否存在,发现存在一个引用与“2b”进行equals相比是相等的,于是返回该引用
System.out.println(s == s1);//true,s就是常量池中“2b”的引用,即s2,而常量池中这个引用由指向s1
//总结:仅在堆中字符串常量池之外创建了一个“2b”字符串对象,字符串常量池存的是它引用,引用变量s也是它的引用。
//我们去掉String s2 = s1.intern();这个语句又如何呢?
int n = 2;//变量
String s1 = n + "b";//只要有一个变量最后拼接出来的字符串对象都是只在堆中常量池之外,常量池中没有。
//String s2 = s1.intern();//没有执行intern方法说明字符串常量池中没有该对象也没有该对象的引用
String s = "2b";//通过引号创建会先去常量池找是否存在,发现不存在,于是在常量池中创建该字符串对象并返回引用
System.out.println(s == s1);//false,s就是常量池中创建的“2b”的引用,而s1中这个引用指向的是常量池之外的“2b”

隐形引号创建

String s = new String("a") + new String("b");//new String("ab")
String s2 = s.intern();//jdk6中,在串池中创建一个字符串“ab”
                       //jdk7/8中,串池中没有创建字符串“ab”,而是创建一个引用,指向new String("ab")
System.out.println(s2=="ab");//jdk6:true  jdk8:true。“ab”表示用引号创建字符串“ab”,去常量池中找发现存在
System.out.println(s=="ab");//jdk6:false  jdk8:true

最后总结

记住三点:

  1. +号拼接两边只要有一个变量最后拼接出来的字符串对象都是只在堆中常量池之外,常量池中没有。
  2. new String()会在堆的字符串常量池之外和池内都创建一个对象
  3. 直接引号创建会只在常量池中创建,而且创建之前会检查常量池中是否存在该内容的字符串对象或引用,存在就返回引用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值