`
seara
  • 浏览: 654949 次
  • 性别: Icon_minigender_1
  • 来自: 广州
社区版块
存档分类
最新评论

在Java中连接字符串时是使用+号还是使用StringBuilder

阅读更多

本文为原创,如需转载,请注明作者和出处,谢谢!

字符串是Java程序中最常用的一种数据结构之一。在Java中的String类已经重载的"+"。也就是说,字符串可以直接使用"+"进行连接,如下面代码所示:

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->Strings="abc"+"ddd";


但这样做真的好吗?当然,这个问题不能简单地回答yes or no。要根据具体情况来定。在Java中提供了一个StringBuilder类(这个类只在J2SE5及以上版本提供,以前的版本使用StringBuffer类),这个类也可以起到"+"的作用。那么我们应该用哪个呢?

下面让我们先看看如下的代码:

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->packagestring;

publicclassTestSimplePlus
{
publicstaticvoidmain(String[]args)
{
Strings
="abc";
Stringss
="ok"+s+"xyz"+5;
System.out.println(ss);
}
}

上面的代码将会输出正确的结果。从表面上看,对字符串和整型使用"+"号并没有什么区别,但事实真的如此吗?下面让我们来看看这段代码的本质。

我们首先使用反编译工具(如jdk带的javap、或jad)将TestSimplePlus反编译成Java Byte Code,其中的奥秘就一目了然了。在本文将使用jad来反编译,命令如下:

jad -o -a -s d.java TestSimplePlus.class

反编译后的代码如下:


<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->packagestring;

importjava.io.PrintStream;

publicclassTestSimplePlus
{
publicTestSimplePlus()
{
//00:aload_0
//11:invokespecial#8<MethodvoidObject()>
//24:return
}

publicstaticvoidmain(Stringargs[])
{
Strings
="abc";
//00:ldc1#16<String"abc">
//12:astore_1
Stringss=(newStringBuilder("ok")).append(s).append("xyz").append(5).toString();
//23:new#18<ClassStringBuilder>
//36:dup
//47:ldc1#20<String"ok">
//59:invokespecial#22<MethodvoidStringBuilder(String)>
//612:aload_1
//713:invokevirtual#25<MethodStringBuilderStringBuilder.append(String)>
//816:ldc1#29<String"xyz">
//918:invokevirtual#25<MethodStringBuilderStringBuilder.append(String)>
//1021:iconst_5
//1122:invokevirtual#31<MethodStringBuilderStringBuilder.append(int)>
//1225:invokevirtual#34<MethodStringStringBuilder.toString()>
//1328:astore_2
System.out.println(ss);
//1429:getstatic#38<FieldPrintStreamSystem.out>
//1532:aload_2
//1633:invokevirtual#44<MethodvoidPrintStream.println(String)>
//1736:return
}
}

读者可能看到上面的Java字节码感到迷糊,不过大家不必担心。本文的目的并不是讲解Java Byte Code,因此,并不用了解具体的字节码的含义。

使用jad反编译的好处之一就是可以同时生成字节码和源代码。这样可以进行对照研究。从上面的代码很容易看出,虽然在源程序中使用了"+",但在编译时仍然将"+"转换成StringBuilder。因此,我们可以得出结论,Java中无论使用何种方式进行字符串连接,实际上都使用的是StringBuilder

那么是不是可以根据这个结论推出使用"+"StringBuilder的效果是一样的呢?这个要从两个方面的解释。如果从运行结果来解释,那么"+"StringBuilder是完全等效的。但如果从运行效率和资源消耗方面看,那它们将存在很大的区别。


当然,如果连接字符串行表达式很简单(如上面的顺序结构),那么"+"StringBuilder基本是一样的,但如果结构比较复杂,如使用循环来连接字符串,那么产生的Java Byte Code就会有很大的区别。先让我们看看如下的代码:


<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->packagestring;

importjava.util.*;

publicclassTestComplexPlus
{
publicstaticvoidmain(String[]args)
{
Strings
="";
Randomrand
=newRandom();
for(inti=0;i<10;i++)
{
s
=s+rand.nextInt(1000)+"";
}
System.out.println(s);
}
}

上面的代码返编译后的Java Byte Code如下:


<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->packagestring;

importjava.io.PrintStream;
importjava.util.Random;

publicclassTestComplexPlus
{

publicTestComplexPlus()
{
//00:aload_0
//11:invokespecial#8<MethodvoidObject()>
//24:return
}

publicstaticvoidmain(Stringargs[])
{
Strings
="";
//00:ldc1#16<String"">
//12:astore_1
Randomrand=newRandom();
//23:new#18<ClassRandom>
//36:dup
//47:invokespecial#20<MethodvoidRandom()>
//510:astore_2
for(inti=0;i<10;i++)
//*611:iconst_0
//*712:istore_3
//*813:goto49
s=(newStringBuilder(String.valueOf(s))).append(rand.nextInt(1000)).append("").toString();
//916:new#21<ClassStringBuilder>
//1019:dup
//1120:aload_1
//1221:invokestatic#23<MethodStringString.valueOf(Object)>
//1324:invokespecial#29<MethodvoidStringBuilder(String)>
//1427:aload_2
//1528:sipush1000
//1631:invokevirtual#32<MethodintRandom.nextInt(int)>
//1734:invokevirtual#36<MethodStringBuilderStringBuilder.append(int)>
//1837:ldc1#40<String"">
//1939:invokevirtual#42<MethodStringBuilderStringBuilder.append(String)>
//2042:invokevirtual#45<MethodStringStringBuilder.toString()>
//2145:astore_1

//2246:iinc31
//2349:iload_3
//2450:bipush10
//2552:icmplt16
System.out.println(s);
//2655:getstatic#49<FieldPrintStreamSystem.out>
//2758:aload_1
//2859:invokevirtual#55<MethodvoidPrintStream.println(String)>
//2962:return
}
}

大家可以看到,虽然编译器将"+"转换成了StringBuilder,但创建StringBuilder对象的位置却在for语句内部。这就意味着每执行一次循环,就会创建一个StringBuilder对象(对于本例来说,是创建了10StringBuilder对象),虽然Java有垃圾回收器,但这个回收器的工作时间是不定的。如果不断产生这样的垃圾,那么仍然会占用大量的资源。解决这个问题的方法就是在程序中直接使用StringBuilder来连接字符串,代码如下:

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->packagestring;

importjava.util.*;

publicclassTestStringBuilder
{
publicstaticvoidmain(String[]args)
{
Strings
="";
Randomrand
=newRandom();
StringBuilderresult
=newStringBuilder();
for(inti=0;i<10;i++)
{
result.append(rand.nextInt(
1000));
result.append(
"");
}
System.out.println(result.toString());
}
}

上面代码反编译后的结果如下:


<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->packagestring;

importjava.io.PrintStream;
importjava.util.Random;

publicclassTestStringBuilder
{

publicTestStringBuilder()
{
//00:aload_0
//11:invokespecial#8<MethodvoidObject()>
//24:return
}

publicstaticvoidmain(Stringargs[])
{
Strings
="";
//00:ldc1#16<String"">
//12:astore_1
Randomrand=newRandom();
//23:new#18<ClassRandom>
//36:dup
//47:invokespecial#20<MethodvoidRandom()>
//510:astore_2
StringBuilderresult=newStringBuilder();
//611:new#21<ClassStringBuilder>
//714:dup
//815:invokespecial#23<MethodvoidStringBuilder()>
//918:astore_3
for(inti=0;i<10;i++)
//*1019:iconst_0
//*1120:istore4
//*1222:goto47
{
result.append(rand.nextInt(
1000));
//1325:aload_3
//1426:aload_2
//1527:sipush1000
//1630:invokevirtual#24<MethodintRandom.nextInt(int)>
//1733:invokevirtual#28<MethodStringBuilderStringBuilder.append(int)>
//1836:pop
result.append("");
//1937:aload_3
//2038:ldc1#32<String"">
//2140:invokevirtual#34<MethodStringBuilderStringBuilder.append(String)>
//2243:pop
}

//2344:iinc41
//2447:iload4
//2549:bipush10
//2651:icmplt25
System.out.println(result.toString());
//2754:getstatic#37<FieldPrintStreamSystem.out>
//2857:aload_3
//2958:invokevirtual#43<MethodStringStringBuilder.toString()>
//3061:invokevirtual#47<MethodvoidPrintStream.println(String)>
//3164:return
}
}

从上面的反编译结果可以看出,创建StringBuilder的代码被放在了for语句外。虽然这样处理在源程序中看起来复杂,但却换来了更高的效率,同时消耗的资源也更少了。

在使用StringBuilder时要注意,尽量不要"+"StringBuilder混着用,否则会创建更多的StringBuilder对象,如下面代码所:

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->for(inti=0;i<10;i++)
{
result.append(rand.nextInt(
1000));
result.append(
"");
}

改成如下形式:

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->for(inti=0;i<10;i++)
{
result.append(rand.nextInt(
1000)+"");
}

则反编译后的结果如下:


<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->
for(inti=0;i<10;i++)
//*1019:iconst_0
//*1120:istore4
//*1222:goto65
{
result.append((
newStringBuilder(String.valueOf(rand.nextInt(1000)))).append("").toString());
//1325:aload_3
//1426:new#21<ClassStringBuilder>
//1529:dup


从上面的代码可以看出,Java编译器将"+"编译成了StringBuilder,这样for语句每循环一次,又创建了一个StringBuilder对象。
如果将上面的代码在JDK1.4下编译,必须将StringBuilder改为StringBuffer,而JDK1.4"+"转换为StringBuffer(因为JDK1.4并没有提供StringBuilder类)。StringBufferStringBuilder的功能基本一样,只是StringBuffer是线程安全的,而StringBuilder不是线程安全的。因此,StringBuilder的效率会更高。






国内最棒的Google Android技术社区(eoeandroid),欢迎访问!

《银河系列原创教程》发布

《Java Web开发速学宝典》出版,欢迎定购

分享到:
评论

相关推荐

    StringBuilder字符串拼接工具

    在处理大量字符串连接时,相比直接使用`+`运算符或`String`对象,`StringBuilder`能提供更高的性能。这是因为每次使用`+`进行字符串连接时,Java都会创建新的`String`对象,这在内存管理和效率上都是不理想的。而`...

    java通用字符串连接

    在Java编程中,字符串连接是常见的操作,尤其是在处理集合数据如List、Set、Map和数组时。`java通用字符串连接`这个主题关注的是如何高效且灵活地将这些集合中的元素合并成一个单一的字符串,同时允许自定义连接分隔...

    java 创建字符串类

    - **字符串连接**:可以使用`+`运算符或`StringBuilder`/`StringBuffer`类来连接字符串。但`+`运算符在处理大量字符串连接时效率较低,因为它会在每次连接时创建新的`String`对象。 2. **StringBuffer与...

    java中常用字符串方法总结

    在Java编程语言中,字符串是极其重要且频繁使用的数据类型。字符串对象的处理涉及很多方法,这些方法使得我们在处理文本时能实现各种功能。本文将深入探讨Java中的一些常用字符串方法,帮助你更好地理解和运用它们。...

    JAVA 字符串 操作

    在Java编程语言中,字符串(String)是一个非常基础且重要的数据类型。它被广泛用于处理文本信息,例如用户输入、文件内容、网络数据等。本文将深入探讨Java中的字符串操作,包括创建、比较、拼接、查找与替换、分割...

    JAVA 字符串应用笔记

    在Java 5及以上版本,可以使用`+`操作符连接字符串,但在大量连接操作时,使用`StringBuilder`或`StringBuffer`更高效。 6. **常量池**: 字符串字面量会被放入常量池,如果两个字符串字面量相同,它们会指向同一...

    java 给字符串编号

    3. **字符串连接**:在Java中,可以使用`+`操作符或者`StringBuilder.append()`方法将编号和字符连接起来。例如,`new StringBuilder().append(i).append(str.charAt(i))`可以在编号和字符之间添加连接。 4. **格式...

    JAVA中处理字符串的类

    在Java编程语言中,处理字符串是一项常见的任务。Java提供了多种内置类来支持字符串操作,但有时这些类可能不能完全满足开发者的特定需求。在这种情况下,开发者可能会选择自定义工具类来封装常用的功能,以提高代码...

    java 插入新的字符串

    `StringBufferDemo_05.java`可能包含一个示例,演示如何在循环中使用`StringBuffer`来构建字符串,从而避免`String`的重复创建: ```java public class StringBufferDemo_05 { public static void main(String[] ...

    连接 字符 串.rar

    - **Java**:在Java中,`+`运算符可以用于连接字符串,但频繁使用会导致性能问题,因为每次连接都会创建新的字符串对象。更高效的方案是使用`StringBuilder`或`StringBuffer`类。 - **JavaScript**:JavaScript的`...

    JAVA的字符串拼接[参照].pdf

    本文主要探讨了在Java中几种不同的字符串拼接方法的性能差异,包括使用操作符`+`、`String.concat()`、`StringBuffer.append()`和`StringBuilder.append()`。以下是对这些方法的详细分析: 1. **字符串拼接操作符 ...

    Java 实例 - 两种连接字符串方法的性能比较源代码-详细教程.zip

    这两种类都是Java中用于构建和连接字符串的高效工具,尤其在循环或大量字符串操作时。 首先,我们来看`StringBuffer`。这个类在Java 5之前是处理字符串连接的主要工具,因为它支持多线程环境。`StringBuffer` 的每...

    Java中字符串的连接和比较.pdf

    Java提供了多种方式来连接字符串: 1. **使用`+`运算符**:这是最直观的方式,`+`运算符可以用于连接字符串,即使其中一个操作数不是字符串,它也会自动转换成字符串。例如: ```java String s = "I am a"; ...

    Java 字符串常用方法

    - `+` 运算符: 可用于连接字符串,例如 `str1 + " " + str2`。 - `concat(String str)`: 直接连接两个字符串。 5. **查找与替换** - `indexOf(String str)`: 查找子字符串第一次出现的位置,找不到则返回-1。 -...

    Java第6章 字符串 含源代码

    - **连接**:可以使用`+`运算符或者`StringBuilder`/`StringBuffer`类来连接字符串。 - **比较**:`equals()`方法用于比较两个字符串的内容是否相同,`compareTo()`或`compareToIgnoreCase()`用于比较字符串的顺序...

    Java字符串连接效率[参照].pdf

    这种做法在循环连接字符串时效率低下,因为每次连接都会生成新的对象。 为了解决这个问题,Java提供了StringBuilder和StringBuffer类。这两个类允许在现有字符串基础上进行追加操作,而不会每次都创建新对象。...

    Java 字符串与文本相关实例源码

    在Java编程语言中,字符串(String)是至关重要的数据类型,用于处理文本信息。字符串是不可变的,这意味着一旦创建,就不能更改其内容。本实例源码集主要关注Java中的字符串和文本处理,提供了多种实际应用的示例,...

    Java 字符串

    `String`类提供了`concat()`方法来连接两个字符串,但在频繁连接操作时,推荐使用`StringBuilder`或`StringBuffer`类,它们在多线程环境下更高效。 六、字符串与字符数组 可以使用`toCharArray()`将字符串转换为...

    Java 字符串操作源码实例集.rar

    使用`+`运算符可以连接字符串,例如`str1 + str2`。但频繁使用可能导致性能问题,因为每次连接都会创建新的字符串对象。 3. **字符串比较**: 可以使用`equals()`方法或`==`运算符来比较字符串内容。`equals()`...

    计算机软件-商业源码-14 连接字符串.zip

    1. **Java**: 在Java中,我们可以使用`+`运算符或者`StringBuilder`类来连接字符串。`+`运算符在处理少量字符串时是可行的,但频繁使用会创建多个中间对象,效率较低。对于大量字符串连接,推荐使用`StringBuilder`...

Global site tag (gtag.js) - Google Analytics