`

String StringBuffer StringBuilder浅析

阅读更多
String类位于java.lang包,继承关系为
public final class String 
    implements java.io.Serializable, Comparable<String>, CharSequence

String内部实现其实就是一个char数组,
/** The value is used for character storage. */
private final char value[];

/** The offset is the first index of the storage that is used. */
private final int offset;

/** The count is the number of characters in the String. */
private final int count;

/** Cache the hash code for the string */
private int hash; // Default to 0


通过offset跟count来控制需要展现出来的字符,但这几个变量也都是final类型,所以String对象一旦创建就不可更改了。

String类提供了大量有用的方法,想着是对char数组进行操作的话,基本就是对数组进行copy,然后通过控制offset及count创建新的String对象,但最后一行有一个方法比较特别
public native String intern();


这个方法的使用涉及到对String对象的存储方式的理解
String在JAVA时是非常特别的一个类,虽然它也提供子大量的构造器方法,但一般来说我们都是通过
String = "abc";
这样的方式来创建一个String对象的。而JVM对String对象的存储方式也不同一般。JVM内存大致可分为堆跟栈,栈存储一些基本数据类型的数据以及对象的引用等,而对象则都保存的堆内存中,堆内存通常可分为JAVA堆跟方法区,方法区内存一般称为永生代,即这部分内存存储的东西基本上是不会被当垃圾回收掉的,是常驻内存。一般的对象是存储在JAVA堆中,但通过上述方法创建的String对象并不是存储在JAVA堆里面,而是存储在方法区的常量区内存中,当JVM创建一个String对象"abc"的时候,并不是直接就开始创建对象而是先在常量区查找有没有"abc"这个对象,如果有则将引用直接指向"abc",如果没有则在常量区创建一个"abc"对象,并将引用指向"abc"的地址。通过下面的代码过证明。
String s1 = "abc";
String s2 = "abc";
String s3 = "a"+"bc";
String s4 = new String("abc");
s1==s2;//true
s2==s3;//true
s1==s4;//false

其中s2==s3完全是编译器的功劳
但如果通过String s = new String("abc")的方式创建String对象则是跟普通的对象一样存储在JAVA堆中,s1==s4为false可以证明他们的存储地址并不一样。
而intern方法则是将JAVA堆里的String对象影射到常量区的一个途径。
String s = "abc";
String t = new String("abc");
s == t.intern();//true
s == t;//false

可以看到使用intern方法会在常量区创建一个与JAVA堆值相同的String对象。

StringBuilder跟StringBuffer都继续自AbstractStringBuilder,AbstractStringBuilder跟String非常像,也是通过内部的一个char数组来实现功能,但这个char数组不是final的,AbstractStringBuilder提供了大量的方法,其中很多方法跟String类都一样,实现方式也一样。AbstractStringBuilder另外提供的最重要的方法就是append跟insert方法。

StringBuilder是线程不安全的,而StringBuffer类的方法大部分都加上了synchronized,可以认为是线程安全的。

关于codePoint以及正则表达式还需要再研究
分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics