您的位置:首页 > 产品设计 > UI/UE

String,StringBuffer,StringBuilder之间的异同

2017-10-25 09:42 405 查看
一,了解String对象

二,StringBuffer对象与String的区别和好处

三,StringBuilder的优点

我们知道String不是八种基本数据类型,那它是怎么存储字符串的?

我们来看一下源码:

public final class String
implements java.io.Serializable, Comparable<String>, CharSequence {}


String是final的,不可被继承。并且它创建的对象一旦被创建就不能被改变了(后面给予代码证明)。

private final char value[];


里面定义了char数组来存储Value,所以我们知道了,String原来是用char来存储字符串的。

我们都知道String可以有两种方法定义:

String str = “”;

String str = new String (“”);

那么你知道String str = new String(“java”);是创建了几个字符创?

是两个,一个是new String(”),一个是“java”,因为在jvm初始化的时候创建了“java”字符串,然后new 使得String构造函数将str的引用指向new出来新字符创(根据“java”构造出的新字符串)。

package testJavaSE;

public class testString {
public static void change(String text) {
text.replace("j", "i");

}
public static void main(String[] args) {
String str = new String("java");
System.out.println(str);
change(str);
System.out.println(str);
}
}


猜猜输出什么?

如果你说输出:

java

iava

那你就错了,应该是输出:

java

java

为什么?就是因为String对象是不可变对象,它的所有方法(在原有的字符串上操作)都会产生一个新的对象,所以我们输出的对象依旧是原来的对象。

4000



虽然改变了“原来的对象”,但是原来的对象没变,而是又生成了一个字符创对象。

我们将上面的代码改进一下:

package testJavaSE;

public class testString2 {
public static String change(String text) {
return text.replace("j","i");

}
public static void main(String[] args) {
String oldstr = new String("java");
System.out.println("old:"+oldstr);
String newstr = change(oldstr);
System.out.println("new:"+newstr);
}

}


这段代码输出的就是:

old:java

new:iava

(具体为什么产生新的对象,参考String方法源码)

二,接下来我们来看看StringBuffer,和String不同StringBuffer是可变的(它的所有属性不是final的),也就是不像String那样在原有的字符创上更改会产生新的字符串。

//StringBuffer的定义
public final class StringBuffer
extends AbstractStringBuilder
implements java.io.Serializable, CharSequence
{...}


我们下面先来看看两段代码的比较:

package testJavaSE;

public class testString3 {
public static void main(String[] args) {
String str="";
String tmp="abcdefghijklmnopqrstuvwxyz";
long startTime = System.currentTimeMillis();
for(int i=0;i<10000;i++) {
str+=tmp;
}
long endTime = System.currentTimeMillis();
System.out.println(endTime-startTime);
}
}
//输出:1705(不同的电脑输出不同)


package testJavaSE;

public class testStringBuffer {
public static void main(String[] args) {
StringBuffer str = new StringBuffer();
String tmp="abcdefghijklmnopqrstuvwxyz";
long startTime = System.currentTimeMillis();
for(int i=0;i<10000;i++) {
str.append(tmp);
}
long endTime = System.currentTimeMillis();
System.out.println(endTime-startTime);
}
}
//输出:2


同样都是添加字符串,StringBuffer的效率是String的1000倍(运算次数越大差距越明显)。就是因为StringBuffer不会创建那么多的对象,而String在改变字符串的同时会浪费大量的时间来创建新的字符串。

//StringBuffer也是由char数组来存储数据的
private transient char[] toStringCache;

//它的初始化大小为16,如果超出容量,有方法对它扩容
public StringBuffer() {
super(16);
}

//它也可以由我们自己定义数组的大小
public StringBuffer(int capacity) {
super(capacity);
}
//在创建一个StringBuffer的时候,它的大小为传进str大小+16
public StringBuffer(String str) {
super(str.length() + 16);
append(str);
}


public synchronized int length() {
return count;
}

@Override
public synchronized int capacity() {
return value.length;
}

@Override
public synchronized void ensureCapacity(int minimumCapacity) {
super.ensureCapacity(minimumCapacity);
}

/**
* @since      1.5
*/
@Override
public synchronized void trimToSize() {
super.trimToSize();
}

/**
* @throws IndexOutOfBoundsException {@inheritDoc}
* @see        #length()
*/
@Override
public synchronized void setLength(int newLength) {
toStringCache = null;
super.setLength(newLength);
}

/**
* @throws IndexOutOfBoundsException {@inheritDoc}
* @see        #length()
*/
@Override
public synchronized char charAt(int index) {
if ((index < 0) || (index >= count))
throw new StringIndexOutOfBoundsException(index);
return value[index];
}
.
.
.
.
.
//它的方法都是带有synchronized关键字的所以它是线程安全的。


总结:String是不可变的,StringBuffer是可变的,且是线程安全的。

三,我们再来看看StringBuilder:

public final class StringBuilder
extends AbstractStringBuilder
implements java.io.Serializable, CharSequence
{..}


这个家伙是不是似曾相识?

它和StingBuffer几乎相同,都继承了相同的父类实现相同的接口。

初始大小都是16,同样可以自动扩容,但它的方法不带有synchronized的关键字,所以如果是单机,或者不涉及到多线程的操作使用StingBuilder效率更高,若设计到多线程和线程安全则建议使用StringBuffer。

效率:

  StringBuilder > StringBuffer > String

  当然这个是相对的,不一定在所有情况下都是这样。

  比如String str = “hello”+ “world”的效率就比 StringBuilder st = new StringBuilder().append(“hello”).append(“world”)要高。

  因此,这三个类是各有利弊,应当根据不同的情况来进行选择使用:

  当字符串相加操作或者改动较少的情况下,建议使用 String str=”hello”这种形式;

  当字符串相加操作较多的情况下,建议使用StringBuilder,如果采用了多线程,则使用StringBuffer。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: