话不多说,直接进入主题。首先是这样一段代码
String s1 = "ab";
String s2 = "ab";
String s3 = "a"+"b";
System.out.println(s1 == s2); // true
System.out.println(s3 == s2); // true
像s1,s2这种字面量显示赋值,Java会直接在字符串常量池中寻找对应的字符串"ab",如果不存在则新建,否则就直接返回常量池中的值。代码String s1 = “ab"时,java会在常量池中开辟一段空间存入字符串"ab”。
接着代码String s2 = “ab”,java同样会在字符串常量池中寻找字符串"ab",此时常量池中存在字符串"a",则直接指向字符串常量池中的"ab"。和s1指向同一个对象。所以结果为true。
同时s3的"a"+“b"也会在java编译时自动优化成"ab”,指向的也是同一个对象。

那如果我们使用new String(“ab”)呢?
String s1 = "ab";
String s2 = new String("ab");
System.out.println(s1 == s2); // false----s2在堆上
首先,要说明new String()一定会在堆内存中开辟一个空间去new 一个对象。且s2一定指向这个堆内存中的对象。但是如果在new String(“ab”)时常量池中不存在"ab",就会现在常量池中创建"ab",在执行new操作。再将new出来的对象引用赋值给s2。这时s1指向的时常量池中的"ab",而s2指向的是堆内存中的对象。如下图:

那如果是字符串拼接时,其中一个值是变量呢?
String s1 = "ab";
String s2 = "a";
String s3 = s2+"b";
System.out.println(s1 == s3); // false
当字符串拼接时其中一个是变量时,java会new 一个StringBuilder对象去append字符串。我们直接看字节码文件:
也就是说s3的值是new StringBuilder().append(s2).append(“b”).toString(),而StringBuilder的toString方法实现也是new String();说白了也是new 一个String对象并返回(这个方法并不会在常量池中创建对应的值)。所以s1 == s3的结果为false;
intern方法
接下来就是String的intern的方法。String的intern方法作用是在常量池中寻找对应的字符串。如果存在就直接指向常量池中的对象,否则就在常量池中新建一个对象,但是这个对象的值是引用String的地址。以下面的代码为例:
String s1 = new String("1") + new String("1");
s1.intern();
String s2 = "11";
System.out.println(s1 == s2); // true
s1同样是通过new StringBuilder().append方法拼接"1"。new String(“1”)在常量池中新建了"1”.但常量池中并不存在"11"。但s1的值是"11",所以s1.intern()时会在常量池中创建一个对象,这个对象存储着s1在堆中的内存地址。此时 String s2 = “11"在常量池中寻找"11”,会指向刚才intern()所创建的"11",也就间接指向了s1。所以s2和s1是同一个对象。
String s = new String("1") + new String("1");
String s2 = "11";
s = s2.intern();
System.out.println(s == s2); // true
如上面的代码,常量池中已经存在"11",调用intern方法时就改变了指针,指向了和s2。
本文详细探讨了Java中字符串的比较,包括字面量赋值、`new String()`、字符串拼接以及`intern()`方法的使用。通过实例分析了不同情况下字符串在内存中的存储位置,解释了`==`为何在不同场景下返回不同的结果,并阐述了`intern()`如何影响字符串对象的引用。
276

被折叠的 条评论
为什么被折叠?



