逛了几个论坛。

不少人在讨论String的“+”,StringBuilder与StringBuffer等一系列的问题。先不多说了了

现分类详述:

1、String的‘+’,底层运行。及效率问题

2、StringBilder与StringBuffer的比較

本篇博文先介绍第一个问题

为了让大家看明确,

我们举例说明吧!

为了加深理解,我们能够来做几个小实验。

javac Test         编译文件

javap -c Test   查看虚拟机指令



实验一:纯字符串

public class Test {
public static void main(String args[]) {
String str = "a";
}
}

// 将字符串 a 存入常数池

   0:   ldc     #2; //String a

   // 将引用存放到 1 号局部变量中

   2:   astore_1

   3:   return

实验二:纯字符串相加

public class Test {
public static void main(String args[]) {
String str = "a" + "b";
}
}

// 将字符串 ab 压入常数池

   0:   ldc     #2; //String ab

   2:   astore_1

   3:   return



实验二能够非常明显地看出,编译器在编译时产生的字节码已经将 "a" + "b" 优化成了 "ab",

同理多个字符串的相加也会被优化处理,须要注意的是字符串常量相加。



实验三:字符串与自己主动提升常量相加

public class Test {
public static void main(String args[]) {
String str = "a" + (1 + 2);
}
}

// 将字符串 a3 压入常数池

   0:   ldc     #2; //String a3

   2:   astore_1

   3:   return



通过虚拟机指令能够看出,1 + 2 自己主动提升后的常量与字符串常量,虚拟机也会对其进行优化。

实验二、实验三结论:常量间的相加并不会引起效率问题



实验四:字符串与变量相加

public class Test {
public static void main(String args[]) {
String s = "b";
String str = "a" + s;
}
}

// 将字符串 b 压入常数池

   0:   ldc     #2; //String b

   // 将引用存放到 1 号局部变量中

   2:   astore_1

   // 检查到很量的相加。这时创建 StringBuilder 对象

   3:   new     #3; //class java/lang/StringBuilder

   // 从栈中复制出数据。即把字符串 b 复制出来

   6:   dup

   // 调用 StringBuilder 的初始构造

   7:   invokespecial   #4; //Method java/lang/StringBuilder."<init>":()V

   // 将字符串 a 压入常数池

   10:  ldc     #5; //String a

   // 调用 StringBuilder 的 append 方法,把字符串 a 加入进去

   12:  invokevirtual   #6; //Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;

   // 从 1 号局部变量中载入数据引用

   15:  aload_1

   // 调用 StringBuilder 的 append 方法,把字符串 b 加入进去

   16:  invokevirtual   #6; //Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;

   // 调用 StringBuilder 的 toString 方法

   19:  invokevirtual   #7; //Method java/lang/StringBuilder.toString:()Ljava/lang/String;

   // 将 toString 的结果保存至 2 号局部变量

   22:  astore_2

   23:  return



实验四能够看出,很量字会串相加时,因为相加的变量中存放的是字符串的地址引用。

由于在编译时无法确切地知道其它详细的值,也就没有办法对其进行优化处理,这时为了

达到连接的效果,其内部採用了 StringBuilder 的机制进行处理(JDK 5 中新增的,我

这里没有 JDK 1.4,预计在 JDK 1.4 下採用的是 StringBuffer),将他们都 append

进去,最后用 toString 输出。

若 s 为其它类型时,比方:int 类型,也是採用同种方式进行处理。



同理。依据实验二的结果,在 String str = "a" + "b" + s; 时。先会优化成 "ab" 再与

s 依据实验四的方式进行处理。这时 StringBuilder 仅调用了两次 append 方法。



假设是 String str = "a" + s + "b"; 这样的形式的就没办法优化了。StringBuilder 得调

用三次 append 方法。



实验四的结论表明,字符串与变量相加时在内部产生了 StringBuilder 对象并採取了一定

的操作。



假设仅仅有一句 String str = "a" + s; 这样子的,其效率与

String str = new StringBuilder().append("a").append(s).toString();

是一样的。



一般所说的 String 採用连接运算符(+)效率低下主要产生在下面的情况中:

public class Test {
public static void main(String args[]) {
String s = null;
for(int i = 0; i < 100; i++) {
s += "a";
}
}
}

每做一次 + 就产生个 StringBuilder 对象,然后 append 后就扔掉。下次循环再到达时重

新产生个 StringBuilder 对象。然后 append 字符串,如此循环直至结束。

假设我们直接採用 StringBuilder 对象进行 append 的话,我们能够节省 N - 1 次创建和

销毁对象的时间。

String的&#39;+&#39;的性能及原理的更多相关文章

  1. warning:deprecated conversion from string constant to &#39;char *&#39;

    warning:deprecated conversion from string constant to 'char *' 解决方式 #include <iostream> using ...

  2. 也谈string.Join和StringBuilder的性能比较

    前几天在园子里面看到一篇讲StringBuilder性能的文章.文章里面给出了一个测试用例,比较StringBuilder.AppendJoin和String.Join的性能.根据该测试结果,&quo ...

  3. Junit 注解 类加载器 .动态代理 jdbc 连接池 DButils 事务 Arraylist Linklist hashset 异常 哈希表的数据结构,存储过程 Map Object String Stringbufere File类 文件过滤器_原理分析 flush方法和close方法 序列号冲突问题

    Junit 注解 3).其它注意事项: 1).@Test运行的方法,不能有形参: 2).@Test运行的方法,不能有返回值: 3).@Test运行的方法,不能是静态方法: 4).在一个类中,可以同时定 ...

  4. List<T>与Dictionary<string,T>频繁检索的性能差距

    一直对LINQ简洁高效的语法青睐有加,对于经常和资料库,SQL语法打交道的C#开发者来说,LINQ无疑是一个非常不错的选择,当要在List<T>(T为一个普通对象)集合中查找满足某些条件的 ...

  5. .Net 中的 string、String、StringBuffer 内存处理性能 和 应用场景

    body { font-family: Segoe UI, SegoeUI, Helvetica Neue, Helvetica, Arial, sans-serif } code { color: ...

  6. 【原创】Java和C#下String类型中的==和equals的原理与区别

    一.Java下 1.几个例子 public static void main(String[] arge) { String str1 = new String("1234"); ...

  7. react+redux渲染性能优化原理

    大家都知道,react的一个痛点就是非父子关系的组件之间的通信,其官方文档对此也并不避讳: For communication between two components that don't ha ...

  8. C风格字符串和C++ string 对象赋值操作的性能比较

    <<C++ Primer>> 第四版 Exercise Section 4.3.1 部分Exercise 4.2.9 习题如下: 在自己本机执行如下程序,记录程序执行时间: # ...

  9. React + Reflux 渲染性能优化原理

    作者:ManfredHu 链接:http://www.manfredhu.com/2016/11/08/23-reactRenderingPrinciple 声明:版权所有,转载请保留本段信息,否则请 ...

随机推荐

  1. BPI-M1P(全志A20)刷Android启动卡之后启动的过程

    http://blog.csdn.net/wb4916/article/details/78031511BPI-M1P(全志A20)刷Android启动卡之后启动的过程 BPI-M1P(全志A20)刷 ...

  2. <a>标签的href、onclick属性

    链接的 onclick 事件被先执行,其次是 href 属性下的动作(页面跳转,或 javascript 伪链接): 参考:https://www.cnblogs.com/happykakeru/ar ...

  3. 生成Nuget 源代码包来重用你的Asp.net MVC代码

    ASP.NET 开发人员有时会陷入一种困境:想要重用以前写过的东西,如一些具有完整功能的Web页面+后台逻辑, 往往不那么直接了当,因此很不爽.经常采用的方式是:找到以前写过的项目,从中挑出来一些有用 ...

  4. glassfish中新建数据源(创建数据库连接池)

    1.浏览器输入:http://localhost:4848 登录glassfish域管理控制台,默认的用户名和密码是amin和adminadmin.(也可以通过NetBeans的服务选项卡--服务器- ...

  5. Java 基础入门随笔(11) JavaSE版——继承、覆盖、抽象类

    1.面向对象的特征二:继承 定义: 指一个对象直接使用另一对象的属性和方法. 继承好处: 1.提供代码的复用性. 2.让类与类直接产生了关系,给第三个特征多态提供了前提. java中支持单继承.不直接 ...

  6. 更新html技术比较

    document.write() document对象的write方法可以很简单的向页面的源代码中添加内容,不过不推荐使用. 优点:可以快速简单的让初学者理解如何向页面添加内容: 缺点: 只有页面初始 ...

  7. bootstrap datatable 数据刷新问题

    在项目中,页面初始化的时候,通过通过向后台请求数据,页面初始化完之后,datatable是有数据的,当我点击页面的搜索按钮(按照时间过滤数据),datatable的数据要能重新刷新或者重载:这一点,我 ...

  8. Jmeter之计数器

    如果需要引用的数据量较大,且要求不能重复或者需要自增,那么可以使用计数器来实现. 计数器(counter):允许用户创建一个在线程组之内都可以被引用的计数器. 计数器允许用户配置一个起点,一个最大值, ...

  9. 使用Unittest做单元测试,addTest()单个case的时候却执行全部的case

    参考: http://tieba.baidu.com/p/6008699660 首先造成这个结果的原因是pycharm配置问题 问题验证: 测试代码: import unittest class Te ...

  10. Java基础(一)--操作符

    Java底层都是使用操作符来操作Java中的数据 常见的操作符:+.-.*./.= 优先级: 当一个表达式存在多个操作符时,操作符的优先级决定了计算顺序,这点在我们刚开始学习数学的时候就会了解到 如果 ...