咱们先对比下String, StringBuffer, StringBuilder这三个类。他们的主要区别通常体如今线程安全和执行效率上。java
类名 | String | StringBuffer | StringBuilder |
---|---|---|---|
线程安全 | 安全(final修饰) | 安全 | 不安全 |
字符串类型 | 常量 | 变量 | 变量 |
执行效率 | 最慢 | 介于两者之间 | 最快 |
String类是用final修饰符修饰的,它的值是不可修改的,所以是线程安全的。 若是一个StringBuffer对象在缓冲区被多个线程使用时,由于StringBuffer的方法都是带有synchronized关键字的,因此能够保证线程安全,而StringBuilder的方法没有该关键字,不能保证线程安全,所以可能会出现一些操做错误。多线程状况下建议使用StringBuffer,单线程建议使用速度较快的StringBuilder。安全
先看一段代码:多线程
String str = "abcdef"; str = str + "123456"; System.out.println(str);
这段代码输出的结果是: “abcdef123456”, 看着好像是str被改变了,但实际上这是一种假象,JVM对上述代码是这样处理的。 1.执行第一行代码:新建一个String对象“abcdef”(该对象保存在字符串常量池中)将“abcdef”对象的实例引用赋值给str(保存在栈中)。 2.执行第二行代码: 再新建一个String对象str,用来执行str + "123456"操做,也就是说,str这个对象是没有发生改变的(String不可变)。每当用String操做字符串时,其实是在不断的建立新的对象,而原来的对象就会变为垃圾被GC回收掉,可想而知这样执行效率会有多低。 一个特殊例子:app
String str = "This is a" + "special" + "example"; StringBuilder stringBuilder = new StringBuilder("This is a").append("special").append("example");
你会发现生成str对象的速度简直太快了,而这个时候StringBuilder速度上根本一点都不占优点。 其实这是JVM的一个把戏,实际上:String str = "This is a" + "special" + "example";
其实就是: String str = “This is a special example”;
因此不须要太多的时间了。 要注意的是,若是你的字符串是来自另外的String对象的话,速度就没那么快了,譬如:ui
String str2 = "This is a"; String str3 = "special"; String str4 = "test"; String str = str2 +str3 + str4;
这时候JVM会规规矩矩的按照原来的方式去作。线程
1.若是要操做少许的数据用 --> String 2.单线程操做字符串缓冲区 下操做大量数据 --> StringBuilder 3.多线程操做字符串缓冲区 下操做大量数据 --> StringBuffercode