前言
经常在面试或者是一些习题上偶尔看到这个函数
却有些懵懵懂懂
今天就通过源代码彻底解析一下
这个代码的主要功能是
返回字符串对象的规范化表示形式
1. 源码
通过查询其源码
翻译过来的主要意思是
返回字符串对象的规范表示。
最初为空的字符串池由String类私有维护。
当调用intern方法时,如果池中已经包含一个与equals(0bject)方法确定的string对象相等的字符串,则返回池中的字符串。
否则,将此String对象添加到池中并返回对该String对象的引用。 接下来,对于任意两个字符串s和t, s.t == t.t intern()当且仅当s.t = (t)为真时为真。
所有字面值字符串和字符串值常量表达式都被实习生。 字符串字面值是在Java TM语言规范的3.10.5节定义的。
返回:具有与此字符串相同内容的字符串,但保证来自于独特的字符串。
最重要的一句话就是
==对于任意两个字符串 s 和 t,当且仅当 s.equals(t) 为 true 时,s.intern() == t.intern() 才为 true。==
通过代码的参数也可得知
返回参数为无
2. 实战代码
通过代码加深这个函数的具体使用规则
public class Test {
public static void main(String args[]) {
String Str1 = new String("https://blog.csdn.net/weixin_47872288?spm=1011.2124.3001.5343");
String Str2 = new String("https://blog.csdn.net/weixin_47872288?spm=1011.2124.3001.5343");
System.out.print("规范表示:" );
System.out.println(Str1.intern());
System.out.print("规范表示:" );
System.out.println(Str2.intern());
System.out.println(Str1==Str1.intern());
}
}
最后输出的结果是
Str1==Str1.intern() 这两者都是输出的一样的值
但是比较的时候却是fasle
是因为采用new 创建的字符串对象不进入字符串池
区分一下这个代码
如果不是使用new string创建对象
String str11 = new StringBuilder("58").append("tongcheng").toString();
System.out.println(str11);
System.out.println(str11.intern());
System.out.println(str11 == str11.intern());
如果将其修改为java
String str11 = new StringBuilder("ja").append("va").toString();
System.out.println(str11);
System.out.println(str11.intern());
System.out.println(str11 == str11.intern());
最主要是因为内部有自带的jdk生成的,之后再new一个字符串,两者进行比较,输出的就是false
==这个坑一定要记住==
再深入探讨挖掘
如果是这个代码
String str1 = "a";
String str2 = "b";
String str3 = "ab";
String str4 = str1 + str2;
String str5 = new String("ab");
System.out.println(str5.equals(str3));//true 引用对比的是内容
System.out.println(str5 == str3);//false 引用等号对比的是地址
System.out.println(str5.intern() == str3);//ture 检查字符串池中是否含有该字符串。由于之前定义的str3已经进入字符串池中,所以会得到相同的引用
System.out.println(str5.intern() == str4);//false 采用new 创建的字符串对象不进入字符串池
结果为
总结:
字符串相加的时候,都是静态字符串的结果会添加到字符串池,如果其中含有变量(如f中的e)则不会进入字符串池中。但是字符串一旦进入字符串池中,就会先查找池中有无此对象。如果有此对象,则让对象引用指向此对象。如果无此对象,则先创建此对象,再让对象引用指向此对象。