【转自http://blog.youkuaiyun.com/flyjimi/article/details/2645063】
例子A:
String str1 = "java";
String str2 = "java";
System.out.print(str1==str2);
地球上有点Java基础的人都知道会输出false,因为==比较的是引用,equals比较的是内容。不是我忽悠大家,你们可以在自己的机子上运行一 下,结果是true!原因很简单,String对象被放进常量池里了,再次出现“java”字符串的时候,JVM很兴奋地把str2的引用也指向了 “java”对象,它认为自己节省了内存开销。不难理解吧 呵呵
例子B:
String str1 = new String("java");
String str2 = new String("java");
System.out.print(str1==str2);
看过上例的都学聪明了,这次肯定会输出true!很不幸,JVM并没有这么做,结果是false。原因很简单,例子A中那种声明的方式确实是在 String常量池创建“java”对象,但是一旦看到new关键字,JVM会在堆中为String分配空间。两者声明方式貌合神离,这也是我把“如何创 建字符串对象”放到后面来讲的原因。大家要沉住气,还有一个例子。
例子C:
String str1 = "java";
String str2 = "blog";
String s = str1+str2;
System.out.print(s=="javablog");
再看这个例子,很多同志不敢妄言是true还是false了吧。爱玩脑筋急转弯的人会说是false吧……恭喜你,你会抢答了!把那个“吧”字去掉你就完 全正确。原因很简单,JVM确实会对型如String str1 = "java"; 的String对象放在字符串常量池里,但是它是在编译时刻那么做的,而String s = str1+str2; 是在运行时刻才能知道(我们当然一眼就看穿了,可是Java必须在运行时才知道的,人脑和电脑的结构不同),也就是说str1+str2是在堆里创建的, s引用当然不可能指向字符串常量池里的对象。没崩溃的人继续看例子D。
例子D:
String s1 = "java";
String s2 = new String("java");
System.out.print(s1.intern()==s2.intern());
intern()是什么东东?反正结果是true。如果没用过这个方法,而且训练有素的程序员会去看JDK文档了。简单点说就是用intern()方法就 可以用“==”比较字符串的内容了。在我看到intern()方法到底有什么用之前,我认为它太多余了。其实我写的这一条也很多余,intern()方法 还存在诸多的问题,如效率、实现上的不统一……
例子E:
String str1 = "java";
String str2 = new String("java");
System.out.print(str1.equals(str2));
无论在常量池还是堆中的对象,用equals()方法比较的就是内容,就这么简单!
-----------------------------------------------------------------------------------------------------------------------------------------------------------------------
String 字符串常量
StringBuffer 字符串变量(线程安全)
StringBuilder 字符串变量(非线程安全)
public class Test {
private int iMaxCount = 100;
/**
* @param args the command line arguments
*/
public static void main(String[] args) {
Test t = new Test();
t.iMaxCount = 10000;
System.out.println("String、StringBuffer、StringBuilder时间比较:" + t.iMaxCount + "次");
t.test1();
t.test2();
t.test3();
System.out.println("------------测试结束-------------");
}
private void test1() {
String str = new String("");
long beginTime = System.currentTimeMillis();
for (int i = 0; i < iMaxCount; i++) {
str += String.valueOf(i);
}
long endTime = System.currentTimeMillis();
System.out.println("String测试时间" + (endTime - beginTime));
}
private void test2() {
StringBuffer sb = new StringBuffer("");
long beginTime = System.currentTimeMillis();
for (int i = 0; i < iMaxCount; i++) {
sb.append(i);
}
long endTime = System.currentTimeMillis();
System.out.println("StringBuffer测试时间" + (endTime - beginTime));
}
private void test3() {
StringBuilder sb = new StringBuilder("");
long beginTime = System.currentTimeMillis();
for (int i = 0; i < iMaxCount; i++) {
sb.append(i);
}
long endTime = System.currentTimeMillis();
System.out.println("StringBuilder测试时间" + (endTime - beginTime));
}
}
结果为:
String、StringBuffer、StringBuilder时间比较:10000次
String测试时间332
StringBuffer测试时间1
StringBuilder测试时间1
------------测试结束-------------
结果很明显了,在字符串拼接的时候最好使用后面两种方法,在使用netbeans开发的时候又建议用StringBuilder替换StringBuffer