java那点事——StringBuffer与StringBuilder原理与区别

原创 2012年10月09日 19:49:53
今天去多玩YY笔试Java开发类职位,这个还是要看看能不能在广州找个好的工作!!

Java类的笔试题中有个简单题是“StringBuffer与StringBuilder的区别,它们的应用场景是什么?”

其实只要找下Google大神就有答案了:StringBuffer 与 StringBuilder 中的方法和功能完全是等价的,只是StringBuffer 中的方法大都采用了 synchronized 关键字进行修饰,因此是线程安全的,而 StringBuilder 没有这个修饰,可以被认为是线程不安全的。 

为了更好的理解上述的答案,还是直接看StringBuffer与StringBuilder的源码实现比较实在,作为一个程序猿,“有疑问,看源码”才是正道,我可以负责任的说,当然了得有条件才行!

jdk的实现中StringBuffer与StringBuilder都继承自AbstractStringBuilder,对于多线程的安全与非安全看到StringBuffer中方法前面的一堆synchronized就大概了解了。

这里随便讲讲AbstractStringBuilder的实现原理:我们知道使用StringBuffer等无非就是为了提高java中字符串连接的效率,因为直接使用+进行字符串连接的话,jvm会创建多个String对象,因此造成一定的开销。AbstractStringBuilder中采用一个char数组来保存需要append的字符串,char数组有一个初始大小,当append的字符串长度超过当前char数组容量时,则对char数组进行动态扩展,也即重新申请一段更大的内存空间,然后将当前char数组拷贝到新的位置,因为重新分配内存并拷贝的开销比较大,所以每次重新申请内存空间都是采用申请大于当前需要的内存空间的方式,这里是2倍。


接下来,玩些好玩的!
在Google中出来了这么一些信息:

    StringBuffer 始于 JDK 1.0 
    StringBuilder 始于 JDK 1.5 

    从 JDK 1.5 开始,带有字符串变量的连接操作(+),JVM 内部采用的是 
    StringBuilder 来实现的,而之前这个操作是采用 StringBuffer 实现的。

我们通过一个简单的程序来看其执行的流程:

清单1 Buffer.java
public class Buffer {
	 public static void main(String[] args) {
	        String s1 = "aaaaa";
	        String s2 = "bbbbb";
	        String r = null;
	        int i = 3694;
	        r = s1 + i + s2; 
			
			for(int j=0;i<10;j++){
				r+="23124";
			}
	 }
}

使用命令javap -c Buffer查看其字节码实现:

清单2 Buffer类字节码


将清单1和清单2对应起来看,清单2的字节码中ldc指令即从常量池中加载“aaaaa”字符串到栈顶,istore_1将“aaaaa”存到变量1中,后面的一样,sipush是将一个短整型常量值(-32768~32767)推送至栈顶,这里是常量“3694”,更多的Java指令集请查看另一篇文章“Java指令集”。

让我们直接看到13,13~17是new了一个StringBuffer对象并调用其初始化方法,20~21则是先通过aload_1将变量1压到栈顶,前面说过变量1放的就是字符串常量“aaaaa”,接着通过指令invokevirtual调用StringBuffer的append方法将“aaaaa”拼接起来,后续的24~30同理。最后在33调用StringBuffer的toString函数获得String结果并通过astore存到变量3中。

看到这里可能有人会说,“既然JVM内部采用了StringBuffer来连接字符串了,那么我们自己就不用用StringBuffer,直接用”+“就行了吧!“。是么?当然不是了。俗话说”存在既有它的理由”,让我们继续看后面的循环对应的字节码。

37~42都是进入for循环前的一些准备工作,37,38是将j置为1。44这里通过if_icmpge将j与10进行比较,如果j大于10则直接跳转到73,也即return语句退出函数;否则进入循环,也即47~66的字节码。这里我们只需看47到51就知道为什么我们要在代码中自己使用StringBuffer来处理字符串的连接了,因为每次执行“+”操作时jvm都要new一个StringBuffer对象来处理字符串的连接,这在涉及很多的字符串连接操作时开销会很大。




StringBuffer详解

1、简介 StringBuffer,由名字可以看出,是一个String的缓冲区,也就是说一个类似于String的字符串缓冲区,和String不同的是,它可以被修改,而且是线程安全的。StringBu...
  • u012877472
  • u012877472
  • 2016年03月05日 16:00
  • 1753

stringbuffer 原理string 区别

stringbuffer  调用append()  通过扩容和copy 一:         expandCapacity(j);  //内部通过copyof构建含有原字符数组的 新的长度...
  • gbk188
  • gbk188
  • 2014年05月13日 17:25
  • 441

string 和bufferstring运用时原理区别

String类用来表示那些创建后就不会再改变的字符串,它是immutable的。 StringBuffer类用来表示内容可变的字符串,并提供了修改底层字符串的方法。 当我们进行字符拼接时,请使用Str...
  • leifuchild
  • leifuchild
  • 2014年02月05日 14:09
  • 1237

为什么String对象不可变,而StringBuffer可变?

1.String对象不可变、StringBuffer对象可变的含义: 举个例子:String str = "aa";  str = "aa"+"bb"; 此时str的值为"aabb",但是"aabb"...
  • qingmengwuhen1
  • qingmengwuhen1
  • 2017年04月06日 18:54
  • 886

String、StringBuilder、 StringBuffer 深入分析 源码解析

java学习有一段时间了,但学习的东西都是框架等东西,java基础知识有点遗忘,所以重温一下java基础知识,写写文章里面有错的希望大家指正共同进步~~ 一、String 大家经常会说使用“+”号...
  • QH_JAVA
  • QH_JAVA
  • 2015年06月06日 11:32
  • 6023

StringBuilder剖析

提到如果字符串修改操作比较频繁,应该采用StringBuilder和StringBuffer类,这两个类的方法基本是完全一样的,它们的实现代码也几乎一样,唯一的不同就在于,StringBuffer是线...
  • qq_17505335
  • qq_17505335
  • 2016年10月13日 13:25
  • 1477

StringBuilder详解

1、简介 StringBuilder和StringBuffer一样,都是继承自抽象类AbstractStringBuilder类,也是一个可变的字符序列。StringBuilder和StringBu...
  • u012877472
  • u012877472
  • 2016年03月06日 11:28
  • 4208

StringBuffer的用法

StringBuffer类和String一样,也用来代表字符串,只是由于StringBuffer的内部实现方式和String不同,所以StringBuffer在进行字符串处理时,不生成新的对象,在内存...
  • randomnet
  • randomnet
  • 2012年04月16日 10:16
  • 22244

从Java源码角度彻底理解String,StringBuilder和StringBuffer的区别

字符串,就是一系列字符的集合。 Java里面提供了String,StringBuffer和StringBuilder三个类来封装字符串,其中StringBuilder类是到jdk 1.5才新增的...
  • hj7jay
  • hj7jay
  • 2016年10月09日 20:31
  • 3542

深入底层代码理解java中String、StringBuffer、StringBuilder

java中String、StringBuffer、StringBuilder是编程中经常使用的字符串类,他们之间的区别也是经常在面试中会问到的问题。现在总结一下,看看他们的不同与相同。 1.可变...
  • sinat_36705710
  • sinat_36705710
  • 2017年03月21日 13:52
  • 1083
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:java那点事——StringBuffer与StringBuilder原理与区别
举报原因:
原因补充:

(最多只允许输入30个字)