String.intern()使用总结
First Blood
先看下面的代码:
String s = new String("1");
String s1 = s.intern();
System.out.println(s == s1);
打印结果为:
false
对于new String("1")
,会生成两个对象,一个是String类型对象,它将存储在Java Heap中,另一个是字符串常量对象"1",它将存储在字符串常量池中。
s.intern()
方法首先会去字符串常量池中查找是否存在字符串常量对象"1",如果存在则返回该对象的地址,如果不存在则在字符串常量池中生成为一个"1"字符串常量对象,并返回该对象的地址。
如下图:
变量s
指向的是Stirng类型对象,变量s1
指向的是"1"字符串常量对象,所以s == s1
结果为false。
Double kill
在上面的基础上我们再定义一个s2如下:
String s = new String("1");
String s1 = s.intern();
String s2 = "1";
System.out.println(s == s1);
System.out.println(s1 == s2); // true
s1 == s2
为true,表示变量s2是直接指向的字符串常量,如下图:
Triple kill
在上面的基础上我们再定义一个t如下:
String s = new String("1");
String t = new String("1");
String s1 = s.intern();
String s2 = "1";
System.out.println(s == s1);
System.out.println(s1 == s2);
System.out.println(s == t); // false
System.out.println(s.intern() == t.intern()); // true
s == t
为false,这个很明显,变量s和变量t指向的是不同的两个String类型的对象。
s.intern() == t.intern()
为true,因为intern方法返回的是字符串常量池中的同一个"1"对象,所以为true。
Ultra kill
在上面的基础上我们再定义一个x和s3如下:
String s = new String("1");
String t = new String("1");
String x = new String("1") + new String("1");
String s1 = s.intern();
String s2 = "1";
String s3 = "11";
System.out.println(s == s1);
System.out.println(s1 == s2);
System.out.println(s == t);
System.out.println(s.intern() == t.intern());
System.out.println(x == s3); // fasle
System.out.println(x.intern() == s3.intern()); // true
变量x为两个String类型的对象相加,因为x != s3
,所以x肯定不是指向的字符串常量,实际上x就是一个String类型的对象,调用x.intern()
方法将返回"11"对应的字符串常量,所以x.intern() == s3.intern()
为true。
Rampage
将上面的代码简化并添加几个变量如下:
String x = new String("1") + new String("1");
String x1 = new String("1") + "1";
String x2 = "1" + "1";
String s3 = "11";
System.out.println(x == s3); // false
System.out.println(x1 == s3); // false
System.out.println(x2 == s3); // true
x == s3
为false表示x指向String类型对象,s3指向字符串常量;
x1 == s3
为false表示x1指向String类型对象,s3指向字符串常量;
x2 == s3
为true表示x2指向字符串常量,s3指向字符串常量;
所以我们可以看到new String("1") + "1"
返回的String类型的对象。
总结
现在我们知道intern方法就是将字符串保存到常量池中,在保存字符串到常量池的过程中会先查看常量池中是否已经存在相等的字符串,如果存在则直接使用该字符串。
所以我们在写业务代码的时候,应该尽量使用字符串常量池中的字符串,比如使用String s = "1";
比使用new String("1");
更节省内存。我们也可以使用String s = 一个String类型的对象.intern();
方法来间接的使用字符串常量,这种做法通常用在你接收到一个String类型的对象而又想节省内存的情况下,当然你完全可以String s = 一个String类型的对象;但是这么用可能会因为变量s的引用而影响String类型对象的垃圾回收。所以我们可以使用intern方法进行优化,但是需要注意的是intern
能节省内存,但是会影响运行速度,因为该方法需要去常量池中查询是否存在某字符串。
参考:https://tech.meituan.com/2014/03/06/in-depth-understanding-string-intern.html