在任何编程语言中,字符串都是我们编写程序时不可避免要用到的常用的数据类型之一. 对于Java初学者而言,当谈到String和StringBuffer的区别时,通常都会有些困惑. 而要弄清楚两者之间的区别,正确地理解String类是最为关键的.
1. java.lang.String is immutable
在Java中string是一个对象,String类位于java.lang包中,在Java中String对象中的每一个字符(character)都是16bit的Unicode字符.
创建一个String对象的方法有如下3种:
1. String s = new String();
s = “abcdef”;
2.String s = new String(“abcdef”);
3.String s = “abcdef”;
上面三种方法的结果是:创建了一个String对象,其值为“abcdef”,将其赋给一个reference variable s.
String is immutable, but it reference is not.
而关于java.lang.String我们要理解的关键一点是:一个String对象一旦被创建,就不允许被改变(Once a String object is created, it can never be created.). 或者我们说,这被称为String objects的immutability,即:一旦赋一个值给String对象,该值就不能够再被改变,这个值是immutable的.
我们要明白的关键是,说String对象是不可变的,是指其值是不可变的,但是该对象的引用变量是可变的,可以将其改变指向任何其他String对象实例(While the String object is immutable, its reference variable is not.). 例如:
String a = “abcedf”;
a2 = a;
a = a.concat(“ more staff”);
上述这个很简单的例子可描述如下:String is immutable.故a的值不能够改变,所以a.concat(“ more staff”)生成了一个新的String:“abcdef more staff”,将String Object reference a重新指向这个新的String Object—“abcdef more staff”。也就是说,迄今为止,一共有三个String Object生成:“abcdef”,“ more staff”以及“abcdef more staff”
必须牢记的是:表面上看到的String的改变,实际上是其reference的改变. 这就体现了String对象的Immutability.
2. Java中关于String的内存管理的特殊性
任何好的编程语言的主要目标之一就是要能够高效地使用宝贵的内存资源.
而在Java中为了高效地使用我们最常用的数据类型:字符串. JVM专门在内存中设定了一个特殊的区域:字符串常量池/String Constant Pool. 当编译器遇到一个字符串字面值/String Leteral,它首先检查字符串常量池/String Constant Pool. 如果池中存在同样的一个字符串,那么编译器就不会再创建新的字符串对象,而是会将一个字符串的引用指向该匹配的字符串对象.
这样,JVM可以保证对于java程序中的字符串字面值/String Literal以较为高效的方式来使用.
但是,我们也应该会注意到,在这种情况下,可能会有多个字符串对象的引用变量指向字符串常量池/String Constant Pool中的同一个字符串对象. 而同时,如果每个指向该对象的引用都对该对象(的值)不加限制的进行改变,就会出现混乱. 所以,在Java中,将String类设置为final的原因之一就是为了保证其immutability特性—没有人可以override String的方法.
特别注意:对于String对象的不当使用,可能会产生许多被丢弃在String Constant Pool中的字符串常量. 会浪费宝贵的内存资源.
3. 哪些String对象可以进入String Constant Pool/字符串常量池
我们需要明确的是:在Java中,JVM只对于String Literal Object才进行池化(把它们放入特殊的内存区域—String Constant Pool)—中. 参看下面的两个例子:
1. String s = "abc"; //创建了一个String对象和一个String类型的reference变量
在这个简单的例子中, “abc”将会被JVM放进String Constant Pool,同时,String类型的reference变量s会指向它
2. String s = new String("abc"); //创建了两个String对象和一个String类型的reference变量
在这个例子中,由于我们使用了new关键字,JVM会创建两个String对象,一个是与例1中相同的String literal Object “abc”,它会被放入到String Constant Poll中去. 而另为一个则使用new关键字创建的普通的Java String对象,它会被放到Java程序所使用的普通得内存区域中去(normal (non-pool) memory).
从上面的叙述可以看到:只有使用“”(没有使用new)直接创建的String literal才可以被放在String constant pool中.
4.常用的String方法
Ø public char charAt(int index)
获得字符串中指定索引处的字符(注意:字符串的索引是从0开始的/String indexes are zero-based)
Ø public String concat(String s)
连接生成一个新的字符串对象,该字符串的值 = 调用者的值 + 做为参数的字符串的值. 例如:
String x = "library";
System.out.println( x + " card"); // output is "library card"
Ø public boolean equalsIgnoreCase(String s)
比较调用者与参数字符串对象的值在不区分大小写的情况下是否相等
Ø public int length()
返回字符串的长度/字符数. 注意与Array对象的属性length区别:String是length()方法/成员函数;Array是length属性/成员变量.
Ø public String replace(char old, char new)
生成发生字符替换后的新字符串. 注意:会替换所有遇到的旧字符.
Ø public String substring(int begin)
返回从参数所指位置—>原字符串结尾处的字串
Ø public String substring(int begin, int end)
返回从参数所指位置之间的字串.
需要注意的是:开始位置的index是zero-based,但是结束位置的则不是. 注意下面例子:
String x = "0123456789";
System.out.println( x.substring(5) ); // output is "56789"
System.out.println( x.substring(5, 8)); // output is "567"
Ø public String toLowerCase()
生成新的全部字符转换为小写的字符串
Ø public String toUpperCase()
生成新的全部字符转换为大写的字符串
Ø public String trim()
生成新的去掉首位空格的字符串
5. StringBuffer
StringBuffer objects are changeable!
StringBuffer对象是可变的,故使用它不会产生很多discard String literal在String constant pool中,相比较String而言,它的效率更高一些.
而且,StringBuffer的许多方法是线程安全的.
6. StrngBuffer的常用方法
Ø public synchronized StringBuffer append(String s)
将参数所示的字符串连接在原StringBuffer所代表的String的末尾。
与String的concat(String)方法不同StringBuffer的append(String)方法不会产生新的对象,只是改变来原对象的值.
Ø public synchronized StringBuffer insert(int offset, String s)
从offset位置开始插入字符串s,其中offset是zero-based的
Ø public synchronized StringBuffer reverse()
反转StringBuffer对象多代表的字符串,例如:
StringBuffer sb = new StringBuffer("A man a plan a canal Panama");
System.out.println( sb ); //output is“amanaP lanac a nalp a nam A”
Ø public String toString()
返回StringBuffer对象所代表的String的值,例如:
StringBuffer sb = new StringBuffer("test string");
System.out.println( sb.toString() ); // output is "test string"