1.从源码角度分析三者是如何存储数据的
String
public final class String
implements java.io.Serializable, Comparable<String>, CharSequence {
/** The value is used for character storage. */
private final char value[];
可以看到String和存储数据的char[],都有final关键字标志,是不可变对象。那String是如何做加法的呢,下面以这个代码为例:
public static void main(String[] args) {
String str = "";
for (int i = 0; i < 100; i++) {
str += "demo ";
}
}
反编译后:
public static void main(java.lang.String[]);
Code:
0: ldc #2 // String
2: astore_1
3: iconst_0
4: istore_2
5: iload_2
6: bipush 100
8: if_icmpge 37
11: new #3 // class java/lang/StringBuilder
14: dup
15: invokespecial #4 // Method java/lang/StringBuilder."<init>":()V
18: aload_1
19: invokevirtual #5 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
22: ldc #6 // String demo
24: invokevirtual #5 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
27: invokevirtual #7 // Method java/lang/StringBuilder.toString:()Ljava/lang/String;
30: astore_1
31: iinc 2, 1
34: goto 5
37: return
从第11行开始,发现字符串相加没加一次会新生成一个StringBuilder对象,这100次循环就要生成100次。如果循环次数再多,不及时回收会造成可观的内存浪费。
StringBuffer
StringBuffer继承AbstractStringBuilder,存储数据的是char[],非final
abstract class AbstractStringBuilder implements Appendable, CharSequence {
/**
* The value is used for character storage.
*/
char[] value;
添加字符串时,append,进行以下操作:
@Override
public synchronized StringBuffer append(String str) {
toStringCache = null;
super.append(str);
return this;
}
可以看到与String直接相加相比,不会生成过多的临时对象,较为省内存。同时,加了“synchronized”关键字保证了append数据的线程安全性。
StringBuilder
StringBuilder继承自AbstractStringBuilder,存储数据也是char[],非final
abstract class AbstractStringBuilder implements Appendable, CharSequence {
/**
* The value is used for character storage.
*/
char[] value;
添加字符串时,append,进行以下操作:
@Override
public StringBuilder append(int i) {
super.append(i);
return this;
}
与StringBuffer类似,不会生成过多的临时对象,较为省内存。但是没有synchronized标识,是非线程安全。
2. 你了解String吗
String a = "hello";
String b = "hello";
String c=new String("hello");
String d=new String("hello");
System.out.println(a==b);
System.out.println(a==c);
System.out.println(c==d);
System.out.println(a==c.intern());
结果是:
true
false
false
true
String是对象,新生成一个对象的时候,地址会不同,因此a!=c c!=d,那为什么会a==b?
原来直接加“”的字符串时放在“常量池”中。变量a和b指向常量池中的同一位置,因此是相等的。而变量c的分配在堆(非常量池)上面的,所以是不相等的。
常量池在JVM内存模型的什么位置?
这个位置和JDK的版本有关。JDK6放在永久带,JDK常量池被分配在堆上。
如何转换成常量string?
String 提供的intern()即可。例如上个代码例子中“a==c.intern()”是相等的。
参考
http://java-performance.info/string-intern-in-java-6-7-8/
网友评论