哎呦,发现了一个总结的不错的blog:https://tech.meituan.com/in_depth_understanding_string_intern.html
JDK API文档中对intern()方法的描述是:返回字符串对象的规范化表示形式。一个初始为空的字符串池,它由类 String 私有地维护。当调用 intern 方法时,如果池已经包含一个等于此 String 对象的字符串(用 equals(Object) 方法确定),则返回池中的字符串。否则,将此 String 对象添加到池中,并返回此 String 对象的引用。
String s = "123"; String s1 = new String("123"); 然而s == s1 么? 答案是不等于的。于是带着这个问题开始了下面的探究.......
这两种初始化的方式有什么区别呢?
1. String s = "123";
首先会先去常量池看看有没有“123”,如果有的话,直接引用常量池对象就可以了,没有的话直接把对象放到常量池
2.String s = new String("123");
如果“123”第一次出现的话,会建两个对象,一个在堆里,一个在常量池里,如果不是第一次出现,那么新建一个在堆里面的对象。
下面来个例子理解理解:
public class Test {
public static void main(String[] args) {
String a = "123";
String Str = new String("123");
String Str1 =new String("123").intern();
System.out.println(a==Str);
System.out.println(a==Str1);
}
}
结果:
false
true
这里好创建了3个对象。
String a = "123";这里创建了1个对象 放在了常量池里。
String str = new String ("123"); 这里在堆里创建了 1个对象.
String Str1 =new String("123").intern(); 在堆中new了一个对象,接着把常量池中有“123”的引用返回来了;
解释:
1.首先String a = "123"; 发现常量池没有“123”,在常量池里添加字符串123
2.String str = new String("123"); 首先在堆里新建一个对象,又发现常量池里已经有“123”了,那就不添加了。
3.String str1 = new String("123").intern(); 查询到常量池里有“123”,直接引用常量池对象.但是仍然会在堆里面创建一个对象,但是没有引用他。
总结:
String的intern()方法就是扩充常量池的一个方法;当一个String实例str调用intern()方法时,Java查找常量池中是否有相同的字符串常量,如果有,则返回其的引用,如果没有,则在常量池中增加一个等于str的字符串并返回它的引用。