Java String的缓冲池详解

来源:互联网 发布:俄勒冈州立大学 知乎 编辑:程序博客网 时间:2024/05/21 11:29
考试中常会碰到String对象构造的题目。例如String s= new String("a");创建了几个对象。

下面根据代码具体解释一下各种String构造情况。

代码如下:
  1. public class TestConstructString {   
  2.     public staticvoid main(String args[])   
  3.     {   
  4.         String s1 = "a";   
  5.         String s2 = "b";   
  6.         String s3 = "ab";   
  7.            
  8.         String s4 = "ab";   
  9.         System.out.println("s3==s4? "+ (s3==s4));   
  10.    
  11.         String s5 = "a"+"b";   
  12.         System.out.println("s3==s5? "+ (s3==s5));   
  13.            
  14.         String s6 = s1+s2;   
  15.         System.out.println("s3==s6? "+ (s3==s6));   
  16.            
  17.         String s7 = new String("ab");   
  18.         System.out.println("s3==s7? "+ (s3==s7));   
  19.            
  20.         final String s8 = "a" ;    
  21.         final String s9 ="b" ;   
  22.         String s10 = s8 + s9;   
  23.         System.out.println("s3==s10? "+ (s3==s10));   
  24.         }   
  25. }  


输出结果为:

  1. s3==s4? true 
  2. s3==s5? true 
  3. s3==s6? false 
  4. s3==s7? false 
  5. 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可以直接使用。

总之:

  1. //情况一 
  2. String s1 = "s"; // 创建了一个对象; 
  3. //情况二 
  4. s1 = new String("s");//创建了两个对象 
  5. //情况三 
  6. String s2 = new String("a" + s1);//创建了三个对象 
0 0