字符串常量池问题
String类的不可变性字符串地址问题String#intern() 池化间接证明字符串常量池的位置new String("hello") 会创建几个对象?new String("abc") + new String("ABC")会创建几个对象
字符串常量池问题 String类的不可变性String类设计为不可变,final修饰String类,表示String类不能被继承;final修饰value数组字段,被赋值后不可改变,同时value字段是私有的,没有对外暴露。
String不可变的优点:
String类型使用最多,参数传递时更加安全。在多线程下绝对安全。字符串在常量池中共享,节约内存,提高效率。
substring/replace/toLowerCase等方法都是在内部再创建一个新的String类对象。
字符串地址问题String s1 = "a"; //串池
String s2 = "b"; //串池
String s3 = "ab"; //串池
String s4 = s1 + s2; //堆区,new StringBuildler().append("a").append("b").toString() -> new String("ab")
String s5 = "a" + "b"; //编译期优化,直接转为“ab”
System.out.println(s3 == s4); //false
System.out.println(s3 == s5); //true
String#intern() 池化
String#Intern()方法会尝试将字符串对象放入串池中,如果串池中存在,则不会放入,如果串池中不存在,则放入,并会把串池中的对象返回。
String s1 = new String("a") + new String("b");
String s2 = s1.intern();
System.out.println(s1 == s2); //true
说明:s1指向堆中的“ab”字符串,执行intern()方法,将“ab”字符串放入串池中,并返回给s2,所以s1和s2都指向同一对象。
String s1 = new String("a") + new String("b");
String s2 = "ab";
s1.intern();
System.out.println(s1 == s2); //false
说明:s1指向堆中的“ab”字符串,s2指向串池中的“ab”字符串,s1执行intern()方法,尝试将“ab”字符串放入串池,但是串池已经存在所不会放入,所以s1仍然指向堆中,s2指向串池。
间接证明字符串常量池的位置在JDK8下设置:-Xmx5m -XX:-UseGCOverheadLimit
Listnew String(“hello”) 会创建几个对象?list = new ArrayList<>(); int i = 0; try { for (int j = 0; j < 260000; j++) { list.add(String.valueOf(j).intern()); i++; } } catch (Throwable e) { e.printStackTrace(); } finally { System.out.println(i); } //抛出异常:java.lang.OutOfMemoryError: Java heap space
查看字节码:
0 new #23 dup 4 ldc #3 6 invokespecial #4 : (Ljava/lang/String;)V> 9 pop 10 return
会创建2个对象:
- 通过new创建一个String对象。类加载完后会在字符串常量池中创建hello字符串实例
0 new #23 dup 4 invokespecial #3 : ()V> 7 new #4 10 dup 11 ldc #5 13 invokespecial #6 : (Ljava/lang/String;)V> 16 invokevirtual #7 19 new #4 22 dup 23 ldc #8 25 invokespecial #6 : (Ljava/lang/String;)V> 28 invokevirtual #7 31 invokevirtual #9 34 astore_1 35 return
会创建6个 对象:
- 对象1:new StringBuilder()对象对象2:new String(“abc”)对象对象3:常量池中的abc字符串实例对象4:new String(“ABC”)对象对象5:常量池中的ABC字符串实例对象6:调用StringBuilder的toString()方法,会创建并返回一个String对象
注意:toString()方法,本质是调用new String()重新创建一个String对象,创建的abcABC字符串不是常量,所以不会放入常量池,String类的本质是char数组。



