一、简单介绍

编写代码过程中,使用"+"和"contact"比较普遍,但是它们都不能满足大数据量的处理,一般情况下有一下四种方法处理字符串拼接,如下:

1、 加号"+"

2、 Stringcontact()方法

3、 StringBufferappend()方法

4、 StringBuilderappend()方法

二、测试代码参考

StringTest

package com.jdk8.event.StringTest;

public class StringTest {
private static final int max = 100000; public static void testPlus() {
System.out.println("************ plusTest() ************"); String str = ""; long start = System.currentTimeMillis(); for (int i = 0; i < max; i++) {
str = str + "a";
} long end = System.currentTimeMillis(); long cost = end - start; System.out.println(" {str + \"a\"} cost=" + cost + " ms");
} public static void testConcat() {
System.out.println("************ concatTest() ************"); String str = ""; long start = System.currentTimeMillis(); for (int i = 0; i < max; i++) {
str = str.concat("a");
} long end = System.currentTimeMillis(); long cost = end - start; System.out.println(" {str.concat(\"a\")} cost=" + cost + " ms");
} public static void testStringBuffer() {
System.out.println("************ StringBufferTest() ************"); long start = System.currentTimeMillis(); StringBuffer strBuffer = new StringBuffer(); for (int i = 0; i < max; i++) {
strBuffer.append("a");
}
strBuffer.toString(); long end = System.currentTimeMillis(); long cost = end - start; System.out.println(" {strBuffer.append(\"a\")} cost=" + cost + " ms");
} public static void testStringBuilder() {
System.out.println("************ StringBuilderTest() ************"); long start = System.currentTimeMillis(); StringBuilder strBuilder = new StringBuilder(); for (int i = 0; i < max; i++) {
strBuilder.append("a");
}
strBuilder.toString(); long end = System.currentTimeMillis(); long cost = end - start; System.out.println(" {strBuilder.append(\"a\")} cost=" + cost + " ms");
}
}

测试类:TestMain

package com.jdk8.event.StringTest;

public class TestMain {
public static void main(String[] args){
System.out.println("执行100次时");
StringTest.testPlus();
StringTest.testConcat();
StringTest.testStringBuffer();
StringTest.testStringBuilder();
}
}

测试结果如下:

执行100次时
************ plusTest() ************
{str + "a"} cost=0 ms
************ concatTest() ************
{str.concat("a")} cost=0 ms
************ StringBufferTest() ************
{strBuffer.append("a")} cost=0 ms
************ StringBuilderTest() ************
{strBuilder.append("a")} cost=0 ms
执行1000次时
************ concatTest() ************
{str.concat("a")} cost=1 ms
************ StringBufferTest() ************
{strBuffer.append("a")} cost=0 ms
************ StringBuilderTest() ************
{strBuilder.append("a")} cost=0 ms
执行10000次时
************ plusTest() ************
{str + "a"} cost=99 ms
************ concatTest() ************
Disconnected from the target VM, address: '127.0.0.1:63818', transport: 'socket'
{str.concat("a")} cost=17 ms
************ StringBufferTest() ************
{strBuffer.append("a")} cost=1 ms
************ StringBuilderTest() ************
{strBuilder.append("a")} cost=0 ms
执行100000次时
************ plusTest() ************
{str + "a"} cost=6176 ms
************ concatTest() ************
{str.concat("a")} cost=1267 ms
************ StringBufferTest() ************
{strBuffer.append("a")} cost=3 ms
************ StringBuilderTest() ************
Disconnected from the target VM, address: '127.0.0.1:63841', transport: 'socket'
{strBuilder.append("a")} cost=2 ms
执行300000次时
************ plusTest() ************
{str + "a"} cost=23554 ms
************ concatTest() ************
{str.concat("a")} cost=17667 ms
************ StringBufferTest() ************
{strBuffer.append("a")} cost=6 ms
************ StringBuilderTest() ************
{strBuilder.append("a")} cost=5 ms
执行500000次时
************ plusTest() ************
{str + "a"} cost=61421 ms
************ concatTest() ************
{str.concat("a")} cost=53173 ms
************ StringBufferTest() ************
{strBuffer.append("a")} cost=9 ms
************ StringBuilderTest() ************
{strBuilder.append("a")} cost=7 ms

1、 方法1 "+"号和方法2 contact()方法适用于数据量比较小的情况下。为了操作代码风格和养成习惯,还是建议尽量不用

2、方法2 StringBufferappend()方法和StringBuilderappend()方法 本质上是一样的,均继承自AbstractStringBuilder抽象类,效率很高,数据量比较大的时候的推荐用法。区别是StringBuffer 是线程安全的,而StringBuilder是线程不安全的。

三、源代码分析

1、 String的"+"方法,编译器对其做了优化,使用StringBuilderappend()方法进行追加,但是每循环一次都会创建一个StringBuilder对象,且都会调用toString()方法转换为字符串,因此开销很大。

注:执行一次String的"+"方法,相当于str = new StringBuilder(str).append("x").toString();

2、调用Stringcontact()方法,其实就是一次数组的拷贝,虽然在内存中的处理都是原子操作,速度很快,但是当返回时是要创建一个新的String对象的,这样不仅限制了效率,更添加了空间的压力,源码如下:

    public String concat(String str) {
int otherLen = str.length();
if (otherLen == 0) {
return this;
}
int len = value.length;
char buf[] = Arrays.copyOf(value, len + otherLen);
str.getChars(buf, len);
return new String(buf, true);
}

3、StringBufferStringBuilderappend()方法均主要使用父类AbstractStringBuilderappend()方法。源码如下:

    public AbstractStringBuilder append(String str) {
if (str == null)
return appendNull();
int len = str.length();
ensureCapacityInternal(count + len);
str.getChars(0, len, value, count);
count += len;
return this;
}

即当数组空间够用的时候,只是在数组后面添加字符或字符串,并不创建新的对象,只是到最后通过toString()方法来生成最终的字符串,效率很快,也很节省空间,是推荐的用法。

参考

https://www.cnblogs.com/twzheng/p/5923642.html

Java 字符串拼接四种方式的性能比较分析的更多相关文章

  1. Java 字符串拼接 五种方法的性能比较分析 从执行100次到90万次

    [请尊重原创版权,如需引用,请注明来源及地址] > 字符串拼接一般使用“+”,但是“+”不能满足大批量数据的处理,Java中有以下五种方法处理字符串拼接,各有优缺点,程序开发应选择合适的方法实现 ...

  2. Java 字符串拼接5种方式性能比较

    https://www.cnblogs.com/twzheng/p/5923642.html

  3. Java基础知识强化25:Java创建对象的四种方式

    1. Java程序中对象的创建有四种方式: ● 调用new语句创建对象,最常见的一种 ● 运用反射手段创建对象,调用java.lang.Class 或者 java.lang.reflect.Const ...

  4. PHP定义字符串的四种方式

    1.简介 在PHP中这门语言中,因为是弱类型语言,因此使用变量时不需提前定义即可使用. 我们在使用php进行开发的时候,大多数使用双引号.单引号进行定义字符串.既然有这两种方式,那么他们之间肯定是有区 ...

  5. C# 拼接字符串的几种方式和性能

    开发过程中常用到的拼接字符串的方法有三种: 1 简单 “+=” 拼接法 1 2 3 4 5 string str="a";   str+="c"+"d ...

  6. 使用位运算、值交换等方式反转java字符串-共四种方法

    在本文中,我们将向您展示几种在Java中将String类型的字符串字母倒序的几种方法. StringBuilder(str).reverse() char[]循环与值交换 byte循环与值交换 apa ...

  7. java 多线程实现四种方式解析Thread,Runnable,Callable,ServiceExcutor,Synchronized ,ReentrantLock

    1.Thread实现: import java.util.Date; import java.text.SimpleDateFormat; public class MyThread extends ...

  8. Java创建线程四种方式

    1.继承Thread类 public class MyThread extends Thread { public MyThread() { } public void run() { for(int ...

  9. java创建对象的四种方式

    1.最常见的 new 一个 2使用反射机制创建对象,直接调用非构造函数 Class obj=Class.forName("A"); A a=obj.newInstance(); C ...

随机推荐

  1. 使用dotenv 管理nodejs 应用的环境变量&&docker-compose 运行

      说明dotenv 是一个很方便的符合12 factor 的环境变量管理工具,使用很方便,实际上里面的代码也不是很多 测试使用docker 进行环境部署,为了方便分发使用pkg 进行打包,使用alp ...

  2. kotlin 编译 运行 hello world

    kotlin 编译器下载地址:https://github.com/JetBrains/kotlin/releases/tag/v1.3.31 解压:kotlin-compiler-1.3.31.zi ...

  3. Django学习笔记之验证和授权

    验证和授权概述 Django有一个内置的授权系统.他用来处理用户.分组.权限以及基于cookie的会话系统.Django的授权系统包括验证和授权两个部分.验证是验证这个用户是否是他声称的人(比如用户名 ...

  4. 关于怎样获取DevExpress GridView过滤后或排序后的数据集问题(转)

    GridView用自带的过滤功能过滤数据后,想要获取过滤后的数据集,有两种方式: 一.笨办法就是循环遍历GridView,根据gridView.GetRow()或者gridView.GetDataRo ...

  5. FileUpload上传

    单文件上传: ASPX: <div> <!-- 文件上传 --> <asp:FileUpload ID="FileUpload1" runat=&qu ...

  6. C# System.Collections

    System.collection类: 动态数组ArrayList 代表了可被单独索引的对象的有序集合.也就是说他是一个动态的数组,你可以通过索引来进行增删改等操作,数组会自动调整数组的大小.允许在列 ...

  7. win2008安装并配置zabbix3.4 agent

    agent 下载地址: https://www.zabbix.com/download_agents 下载解压后有2个文件夹,一个是bin,存放32位和64位执行程序,一个是conf目录存放着配置文件 ...

  8. EF提示“序列化类型为XXX的对象时检测到循环引用”

    能看到这个标题 ,我就用解释原因,网上很多,我只给大家一个解决方案会: public <#=code.Escape(entity)#> ToPOCO() { return new < ...

  9. Webpack 模块处理

    webpack模块处理 1. ES6 静态Import ES6的import会被转化为commonjs格式或者是AMD格式,babel默认会把ES6的模块转化为commonjs规范的. import ...

  10. LeetCode——295. Find Median from Data Stream

    一.题目链接: https://leetcode.com/problems/find-median-from-data-stream 二.题目大意: 给定一段数据流,要求求出数据流中的中位数,其中数据 ...