java String 缓冲池概念的举例说明
口试中常会碰到String对象构造的题目。例如String s= new String("a");创建了几个对象。
下面根据代码具体解释一下各种String构造情况。
代码如下:
public class TestConstructString { public static void main(String args[]) { String s1 = "a"; String s2 = "b"; String s3 = "ab"; String s4 = "ab"; System.out.println("s3==s4? "+ (s3==s4)); String s5 = "a"+"b"; System.out.println("s3==s5? "+ (s3==s5)); String s6 = s1+s2; System.out.println("s3==s6? "+ (s3==s6)); String s7 = new String("ab"); System.out.println("s3==s7? "+ (s3==s7)); final String s8 = "a" ; final String s9 = "b" ; String s10 = s8 + s9; System.out.println("s3==s10? "+ (s3==s10)); } }
输出结果为:
s3==s4? true s3==s5? true s3==s6? false s3==s7? false s3==s10? true
在解释之前先对String做个简单的先容。
百度百科:java中的字符串。 String类是不可变的,对String类的任何改变,都是返回一个新的String类对象。 String 对象是 System.Char 对象的有序集合,用于表示字符串。String 对象的值是该有序集合的内容,并且该值是不可变的。
特别留意:String类是不可变(final)的,对String类的任何改变,都是返回一个新的String类对象.这样的话把String类的引用传递给一个方法,该方法对String的任何改变,对原引用指向的对象没有任何影响,这一点和基本数据类型相似.
String池:String是不可改变的,为了进步效率Java引用了字符串池的概念,例如new String("abc");首先会在String池中创建一个对象“abc”由于有NEW的 存在所以会分配地址空间copyString池的内容。当出现的String对象在String池中不存在时即在String池中创建该对象。
s3与s4根据String的概念他们都指向了同一个缓冲池内的地址,所以结果为true
s3与s5由于相加的两个为常量所以编译器会把s5="a"+"b"优化为s5="ab"。所以结果也为true。
s3与s6由于是两个变量的相加所以编译器无法优化,s1+s2即等同于(new StringBuilder(String.valueOf(s1))).append(s2).toString(); 在运行时,会有新的String地址空间的分配,而不是指向缓冲池中的“ab”。所以结果false。
s3与s7,根据缓冲池的定义在new的时候实际会新分配地址空间,s7指向的是新分配的地址空间所以与缓冲池地址不同,所以为false
s3与s10,类似于s3与s5,由于是final类型编译器进行了优化所以相同。
创建字符串的方式很多,回纳起来有三类:
其一,使用new关键字创建字符串,比如String s1 = new String("abc");
其二,直接指定。比如String s2 = "abc";
其三,使用串联天生新的字符串。比如String s3 = "ab" + "c";
String对象的创建
String对象的创建也很讲究,关键是要明白其原理。
原理1:当使用任何方式来创建一个字符串对象s时,Java运行时(运行中JVM)会拿着这个X在String池中找是否存在内容相同的字符串对象,假如不存在,则在池中创建一个字符串s,否则,不在池中添加。
原理2:Java中,只要使用new关键字来创建对象,则一定会(在堆区或栈区)创建一个新的对象。
原理3:使用直接指定或者使用纯字符串串联来创建String对象,则仅仅会检查维护String池中的字符串,池中没有就在池中创建一个,有则罢了!但尽不会在堆栈区再往创建该String对象。
原理4:使用包含变量的表达式来创建String对象,则不仅会检查维护String池,而且还会在堆栈区创建一个String对象。
另外,String的intern()方法是一个本地方法,定义为public native String intern(); intern()方法的价值在于让开发者能将留意力集中到String池上。当调用 intern 方法时,假如池已经包含一个即是此 String 对象的字符串(该对象由 equals(Object) 方法确定),则返回池中的字符串。否则,将此 String 对象添加到池中,并且返回此 String 对象的引用。
最后,有几点题目请大家留意:String a; 与String a=null在作为类变量时候是等价的,在局部变量则不同。null表示一个空引用,String a=null意思是在栈中声明了a,但是这个a没有指向任何地址。此时我们留意到String a 在栈中声明了a,但是也没有指向任何地址,但是java的语法检查假如在局部变量中,String a;是不能直接使用的,String a=null中的这个a可以直接使用。
总之:
//情况一 String s1 = "s"; // 创建了一个对象; //情况二 s1 = new String("s"); //创建了两个对象 //情况三 String s2 = new String("a" + s1); //创建了三个对象