String,StringBuffer,StringBuilder之间的异同

一,了解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对象是不可变对象,它的所有方法(在原有的字符串上操作)都会产生一个新的对象,所以我们输出的对象依旧是原来的对象。
这里写图片描述

虽然改变了“原来的对象”,但是原来的对象没变,而是又生成了一个字符创对象。
我们将上面的代码改进一下:

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。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值