`

String对象操作符“+”解析

    博客分类:
  • J2SE
阅读更多
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 次创建和
销毁对象的时间。
分享到:
评论

相关推荐

    JavaScript 布尔操作符解析 || !

    该操作符首先会将它的操作数转换为一个布尔值,然后再对其求反。 下面说明下Boolean()转型函数的一组规则。 数据类型 转换为true的值 转换为false的值 Boolean true false String  任何非空字符串 “”(空...

    day023-xml解析笔记和代码.rar

    名称字符之间不能有空格或者制表符; 例如&lt;四川 省&gt; 12.名称字符之间不能使用冒号;&lt;xml:xx&gt;&lt;/xml: xx&gt; 注意: 1.Xml中空格和换行都表示数据,严格区分大小写 1.4 XML约束:(了解) 1. ...

    Android中Java instanceof关键字全面解析

    instanceof关键字用于判断一个引用类型变量所指向的对象是否是一个类(或接口、抽象类、父类)的实例。... 我们声明了一个String对象引用,指向一个String对象,然后用instancof来测试它所指向的对象是否是Obje

    C#5.0本质论第四版(因文件较大传的是百度网盘地址)

    3.5.1 关系操作符和相等性操作符 77 3.5.2 逻辑布尔操作符 77 3.5.3 逻辑求反操作符 78 3.5.4 条件操作符 79 3.5.5 空接合操作符 80 3.6 按位操作符 80 3.6.1 移位操作符 81 3.6.2 按...

    json的介绍 以及 json2.js的用法

    或者下标操作符检索。 myJSONObject.bindings[0].method // "newURI" 1. 使用eval&#40;&#41;函数: 为了将JSON文本转换为对象,可以使用eval&#40;&#41;函数。eval&#40;&#41;函数调用JavaScript编辑器。由于 JSON是...

    一个java正则表达式工具类源代码.zip(内含Regexp.java文件)

    逗号, 帽号: 数学减号- 右尖括号&gt; 左尖括号反斜杠\ 即空格,制表符,回车符等 10 匹配非负整数(正整数 + 0) 11 匹配不包括零的非负整数(正整数 &gt; 0) 12 匹配正整数 13 匹配非正整数(负整数 + 0) 14 匹配负...

    Android静默安装常用工具类

    parseKeyAndValueToMap(String source, String keyAndValueSeparator, String keyAndValuePairSeparator, boolean ignoreSpace) 字符串解析为map toJson(Map&lt;String, String&gt; map) map转换为json格式 源码可见...

    C++new和delete详细解析

    C++中的new 和delete 是操作符,new 分配对象时候会自动调用构造函数,delete对象时候,会自动调用析构函数,而C语言中 malloc() 和 free() 是函数。 理论上malloc free 和 new 、delete 可以混搭用,但是好不要...

    java常用工具类的使用

    调用DateFormat对象的format方法可以把Date对象转换成为指定格式的String类型数据。比如: Date today=new Date(); DateFormat df=DateFormat.getDateInstance(DateFormat.FULL,Locale.CHINA); String result=df....

    深入java虚拟机第二版

    深入java虚拟机第二版 第1章 Java体系结构介绍 1.1 为什么使用Java 1.2 网络带来的挑战和机遇 ...附录A 按操作码助记符排列的指令集 附录B 按功能排列的操作码助记符 附录C 按操作码字节值排列的操作码助

    深入JAVA虚拟机(第2版)

    前言 第1章 Java体系结构介绍 1.1 为什么使用Java 1.2 网络带来的挑战和机遇 1.3 体系结构 1.3.1 Java虚拟机 ...附录C 按操作码字节值排列的操作码助记符 附录D Java虚拟机的一个模拟:“Slices of Pi”

    深入Java虚拟机

    第1章 Java体系结构介绍 1.1 为什么使用Java ...附录A 按操作码助记符排列的指令集 附录B 按功能排列的操作码助记符 附录C 按操作码字节值排列的操作码助 记符 附录D Java虚拟机的一个模拟:“Slices of Pi”

    C++入门到精通

    这一章也介绍并解释了函数对象的概念 函数对象使我们能够为泛型算法中用到的操作符 比 如等于或小于操作符 提供另一种可替换的语义 关于泛型算法在附录中有详细说明 并带 有用法的示例 第 13 章到第 16 章 第四...

    JavaScript王者归来part.1 总数2

     4.3 表达式和运算符符   4.4 控制语句 句   4.5 总结   第5章 数据类型  5.1 基本数据类型   5.2 数组和对象   5.2.1 数组   5.2.2 对象--一个构造函数的例子   5.3 函数类型--一个函数和闭包的...

    深入Java虚拟机(原书第2版).pdf【附光盘内容】

    《深入Java虚拟机(原书第2版)》,原书名《Inside...附录a 按操作码助记符排列的指令集 附录b 按功能排列的操作码助记符 附录c 按操作码字节值排列的操作码助 记符 附录d java虚拟机的一个模拟:“slices of pi”

    freemarker总结

    options:该参数可以省略,指定包含时的选项,包含encoding和parse两个选项,其中encoding指定包含页面时所用的解码集,而parse指定被包含文件是否作为FTL文件来解析,如果省略了parse选项值,则该选项默认是true. ...

    java基础案例与开发详解案例源码全

    9.1.2 字符串对象操作224 9.1.3 字符串对象修改228 9.1.4 类型转换230 9.2 StringBuffer类的使用231 9.3 StringBuilder类的使用233 9.4 日期类简介234 9.5 Java语言国际化时间获取与计算238 9.6 Random类和Math类240...

    Java语言基础下载

    文档对象模型(DOM)解析实例 402 DOM对象 404 DOM解析的例子: 406 SAX解析实例 409 DOM4J解析实例 412 JDOM解析实例 413 JAVA操纵XML 实例讲解 414 通过JAVA写数据到XML里面 415 内容总结 418 独立实践 418 第二十...

    java 面试题 总结

    JAVA平台提供了两个类:String和StringBuffer,它们可以储存和操作字符串,即包含多个字符的字符数据。这个String类提供了数值不可改变的字符串。而这个StringBuffer类提供的字符串进行修改。当你知道字符数据要改变...

    C++网络爬虫项目

    取服务器的IP地址,构造DnsUrl对象压入解析统一资源定位符队列。 2.3.2. 发送线程(SendThread) 通过WebCrawler对象启动新的抓取任务,从解析统一资源定位符队列中弹出 DnsUrl对象,向HTTP服务器发送HTTP请求,并将套...

Global site tag (gtag.js) - Google Analytics