String、StringBuilder和StringBuffer的区别?

本文解析了Java中String的不可变性,以及StringBuilder和StringBuffer的可变性及线程安全性。强调了在for循环中避免使用+拼接字符串以提高性能,推荐使用StringBuilder的append方法。
摘要由CSDN通过智能技术生成

回答:

String是不可变的,String Builder2和StringBuffer是可变的。而StringBuffer,是线程安全的,而StringBuilder是非线程安全的。

String的“+”是如何实现的

使用+拼接字符串,其实只是Jva提供的一个语法糖,那么,我们就来解一解这个语法糖,看看他的内部原理到底是如何实现的。


还是这样一段代码。我们把他生成的字节码进行反编译,看看结果。

String wechat = "xiaolin";
String introduce = "introduce";
String name= wechat +"," + introduce;

反编译后的内容如下,反编译工具为jad。

string wechat  = "xiaolin";
string introduce = "introduce";
String name= (new StringBuilder()).append(wechat).append(",").append(introduce).toString();

通过查看反编译以后的代码,我们可以发现,原来字符串常量在拼接过程中,是将Stig转成了
String Builder,后,使用其append方法进行处理的。          

        
那么也就是说,Java中的+对字符串的拼接,其实现原理是使用StringBuilder..append。 

StringBuffer和StringBuilder

接下来我们看看StringBuffer和StringBuilder的实现原理。

和String类似,StringBuilder类也封装了一个字符数组,定义如下:

char[] value;

与String不同的是,它并不是final的,所以他是可以修改的。另外,与String不同,字符数组中
不一定所有位置都已经被使用,它有一个实例变量,表示数组中已经使用的字符个数,定义如下:

int count;

其append源码如下:

public stringBuilder append(String str){
    super.append(str);
    return this;
}

该类继承了`AbstractStringBuilder'类,看下其`append方法:

public AbstractstringBuilder append(String str){
    if (str =null)
        return appendNull();
    int len = str.length();
    ensureCapacityInternal(count len);
    str.getchars(0,len,value,count);
    count += len;
    return this;
}

append会直接拷贝字符到内部的字符数组中,如果字符数组长度不够,会进行扩展。


StringBuffer和StringBuilder类似,最大的区别就是StringBuffer是线程安全的,看一下StringBuffer的append方法。

public synchronized stringBuffer append(String str){
    toStringcache null;
    super.append(str);
    return this;
}

该方法使用synchronized进行声明,说明是一个线程安全的方法。而StringBuilder则不是线程安全的。

不要在for循环使用+拼接字符串

前面我们分析过,其实使用+拼接字符串的实现原理也是使用的StringBuilder,那为什么不建议大家在for循环中使用呢?

// 我们把以下代码反编译下:
long t1 = System.currentTimeMillis();
String str = "xiaolin";
for(inti=0;i<50000;i+){
    String s = String.valueof(i);
    str += s;
}
long t2 = System.currentTimeMillis();
System.out.println("+cost:" + (t2-t1));

反编译之后:

// 我们把以下代码反编译下:
long t1 = System.currentTimeMillis();
String str = "xiaolin";
for(inti=0;i<50000;i+){
    String s = String.valueof(i);
    str = (new StringBuilder()).append(str).append(s).toString();
}
long t2 = System.currentTimeMillis();
System.out.println(  (new StringBuilder()).append("+cost:").append(t2-t1).toString());

我们可以看到,反编译后的代码,在for循环中,每次都是new了一个StringBuilder,然后再把
StringBuilder转成String,再进行append。

而频繁的新建对象当然要耗费很多时间了,不仅仅会耗费时间,频繁的创建对象,还会造成内存资源的浪费。

所以,阿里巴巴Jva开发手册建议:循环体内,字符串的连接方式,使用StringBuilder的
append方法进行扩展。而不要直接使用+。

  • 4
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值