java字符串常量池
string
1直接给string赋值
string是一个不可变的字符序列,无论什么时候都不可改变,
value地址不可变,而且value里面的值也不可以变(应该是底层设置的)。
String s1 = "abc";
String s2 = "abc";
s1 = "Hello"
System.out.println(s1 == s2);
"abc"字符串保存在方法区的常量池中,其中方法区的常量池不会保存相同的字符串。s1和s2指向的是字符串常量池的同一个地址。
s1 = "hello"
执行这条语句后s1指向新的地址。
s3 = "abc" + "hello"
执行这条语句后,会在字符串常量池新开辟一块区域放"abchello",s3会指向这个区域
replace,insert,delete等语句都不会对字符串常量池的值进行更改,而是再字符串常量池中开辟一块新的区域存放新值。
2 string new一个对象
new的時候会在string会指向堆,并开辟一个对象,对象的value属性指向常量池中的字符串
String s1 = "javaEE";
String s2 = "javaEE";
String s3 = new String("javaEE");
String s4 = new String("javaEE");
System.out.println(s1 == s2); // true
System.out.println(s3 == s4); // false
System.out.println(s1 == s3); // false
3 字符串拼接
拼接的时候,只要有变量参与(如s5 s6 s7),就都看成是在堆中new了一个对象
String s1 = "javaEE";
String s2 = "hadoop";
String s3 = "javaEEhadoop";
String s4 = "javaEE" + "hadoop";
String s5 = s1 + "hadoop";
String s6 = "javaEE" + s2;
String s7 = s1 + s2;
System.out.println(s3 == s4);//true
System.out.println(s3 == s5);//false
System.out.println(s3 == s6);//false
System.out.println(s3 == s6);//false
System.out.println(s3 == s7);//false
调用intern()会返回常量池中的地址。
jdk有规范,但是不同的jvm有不同的结构,如安卓的jvm比较轻量。从规范上来讲,方法区(永久区)应该在堆中,但事实上方法区都不放在堆中。
字符串常量池在jdk6和jdk8中都放在方法区中,jdk7中放在堆里。
常量和常量的拼接结果在常量池中。如下
final String s = "javaEE";
String s1 = s + "hadoop";
这个时候s1指向的是字符串常量池而不是堆中。
stringbuffer
可变字符序列,线程安全(里面方法好多都是同步方法)
stringbuilder
可变字符序列,线程不安全(里面方法好多都是同步方法),但是提高了效率。