JAVA虚拟机有一个字符串池,对于字符串池的访问可以使用字符串对象的intern()方法,可动态向池中添加对象,它的定义如下:
public native String intern();
这是一个本地方法,在调用这个方法时,JAVA虚拟机首先检查字符串池中是否存在与该字符串对象值相等的对象,如果存在就返回字符串池中的对象的引用,否则就新创建一个与该字符串对象值相等的对象,然后再返回新创建对象的引用,有如下一段代码:
public class InternTest{
public static void main(String args[]){
//用一个字符数组初始化a,是避免池中含有值为"abcd"的对象
String a=new String(new char[]{'a','b','c','d'});
String b=a.intern();
(1)
if(a==b){
System.out.println("b加入了字符串池中,没有新建对象");
}else{
System.out.println("b没加入字符串池中,新建了对象");
}
}
}
上述代码运行后将输出"b没加入字符串池中,新建了对象".原因如下:
String a=new String(new char[]{'a','b','c','d'});
该语句会新建一个字符串对象,但该对象并不会放入字符串池中,而是放在堆中
通过a.intern()语句,JVM将会在字符串池中新建一个值为"abcd"的对象,并把其引用返回给b,因此才有了上面的输出
如果在代码中(1)处加入语句:String c="abcd";
那么System.out.println(c==b);会返回true
前面谈到String对象的存储,在JAVA中,栈(stack)主要保存基本类型,速度仅次于寄存器,比堆要快,而堆(heap)主要用于存储对象;
在用引号创建字符串对象时,实际上是JAVA虚拟机在栈中创建多个char型字符,然后在堆中创建一个String对象,该对象的值就是之前栈中的那个字符序列;而用new String("abc")创建String对象时,关键字new会创建一个字符串对象放在堆中,"abc"也会创建一个字符串对象,但该对象是放在字符串池中的,前者是在运行期创建,后者在编译期创建.因此创建了两个字符串对象,而且两个对象的值都是abc