热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

C#:字符串连接(+=)与StringBuilder效率分析

在C#或Java中,我们可以使用诸如StringBuilder,StringBuffer等方式对大量字符串进行拼接操作。当然,我们也可以直接使用字符串连

在 C# 或 Java 中,我们可以使用诸如 StringBuilder, StringBuffer 等方式对大量字符串进行拼接操作。当然,我们也可以直接使用 字符串连接 (+=) 的方式进行拼接操作。但是,两种主要方式在效率上有何区别呢?我们是否需要针对这两类方式进行区别场景的使用呢?针对于此,我做了如下测试,以证明两种不同方式连接字符串的效率临界值。(关于Java中StringBuilder和StringBuffer的区别,可以参阅这篇文章)

测试方法是:重复连接字符串 “Hello World!” 1次到40次,在每次连接的过程中都分别采用 String Connection (+=) 和 StringBuilder 的 Append()方法进行1000次,并分别计算出两种方法在这1000次运算过程中的获胜比率。通过多次试验,得到如下统计:

第一次:
StringBuilder Efficiency Test 1

第二次:
StringBuilder Efficiency Test 2

第三次:
StringBuilder Efficiency Test 3

多次测试结果显示,在少于8~9次字符串连接时,String Connection (+=) 的效率要略胜一筹。这个结果与 Alois Kraus 给出的测试结果基本一致。

So, 该啥时候使用 +=,想必大家可以了解了。在叠加次数已知的前提下,若少于10次,可以优先考虑 +=,否则请不要犹豫的使用StringBuilder。

* 这里需要注意的是,StringBuilder 的默认 Capacity 是 16,当然,测试结果还没到这个数值,所以不会在较大程度上受此影响。
internal const int DefaultCapacity = 16;

相关测试代码如下:
[csharp]
namespace StringEfficiencyTest
{
#region using directives

using System;
using System.Diagnostics;
using System.IO;
using System.Text;

#endregion using directives

internal class Program
{
private static readonly Stopwatch stopwatch = new Stopwatch();
private static readonly string resultFilePath = @"D:result.txt";

private static void Main()
{
try
{
LogFile("TimestStringConnectionWinRatetStringBuilderWinRate");
const int accuracyCorrection = 1000;

for (var innerTimes &#61; 1; innerTimes <&#61; 40; innerTimes&#43;&#43;)
{
var result &#61; CalculateOnce(innerTimes, accuracyCorrection);
Console.WriteLine("Calculate Times: {0} {1}", innerTimes, result ? string.Empty : "&#64;&#64;&#64;&#64;&#64;&#64;&#64;&#64;&#64;&#64;&#64;&#64;");
}

Console.ReadKey();
}
catch (Exception ex)
{
Console.WriteLine(ex);
}
}

// If stringConnection win return true
private static bool CalculateOnce(int times, int accuracyCorrection)
{
var stringConnectionWinCounter &#61; 0;
var stringBuilderWinCounter &#61; 0;
for (var i &#61; 0; i {
//Console.WriteLine("Stacking fold: " &#43; times);

stopwatch.Reset();
var stringConnectionUsedTime &#61; StringConnectionTest(times);
//Console.WriteLine("String Connection: " &#43; stringConnectionUsedTime);

stopwatch.Reset();
var stringBuilderUsedTime &#61; StringBuilderTest(times);
//Console.WriteLine("String Builder: " &#43; stringBuilderUsedTime);

if (stringConnectionUsedTime {
stringConnectionWinCounter&#43;&#43;;
}

if (stringConnectionUsedTime > stringBuilderUsedTime)
{
stringBuilderWinCounter&#43;&#43;;
}
}

Console.WriteLine("---------------------");
var stringConnectionWinRate &#61; stringConnectionWinCounter*100.0/
(stringConnectionWinCounter &#43; stringBuilderWinCounter);
Console.WriteLine("String Connection Win Counter: {0}, rate: {1:0.0000}%.", stringConnectionWinCounter,
stringConnectionWinRate);
var stringBuilderWinRate &#61; stringBuilderWinCounter*100.0/
(stringConnectionWinCounter &#43; stringBuilderWinCounter);
Console.WriteLine("String Builder Win Counter: {0}, rate: {1:0.0000}%.", stringBuilderWinCounter,
stringBuilderWinRate);

Console.WriteLine("&#61;&#61;# {0} win. #&#61;&#61;",
stringConnectionWinRate > stringBuilderWinCounter ? "String Connection" : "String Builder");

LogFile(string.Format("{0}t{1}%t{2}%", times, stringConnectionWinRate, stringBuilderWinRate));

return stringConnectionWinCounter > stringBuilderWinCounter;
}

private static long StringConnectionTest(int times)
{
var str &#61; string.Empty;

stopwatch.Start();

for (var i &#61; 0; i {
str &#43;&#61; "Hello World!";
}

stopwatch.Stop();

return stopwatch.ElapsedTicks;
}

private static long StringBuilderTest(int times)
{
// internal const int DefaultCapacity &#61; 16;
var str &#61; new StringBuilder();
//var str &#61; new StringBuilder(times);

stopwatch.Start();

for (var i &#61; 0; i {
str.Append("Hello World!");
}

stopwatch.Stop();

return stopwatch.ElapsedTicks;
}

private static void LogFile(string content)
{
using (var fileStream &#61; File.Open(resultFilePath, FileMode.Append))
{
var writer &#61; new StreamWriter(fileStream);
writer.WriteLine(content);
writer.Flush();
}
}
}
}
[/csharp]

查看原文&#xff1a;http://nap7.com/me/stringbuilder-and-stringconnection/


推荐阅读
  • ALTERTABLE通过更改、添加、除去列和约束,或者通过启用或禁用约束和触发器来更改表的定义。语法ALTERTABLEtable{[ALTERCOLUMNcolu ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
  • 本文整理了Java面试中常见的问题及相关概念的解析,包括HashMap中为什么重写equals还要重写hashcode、map的分类和常见情况、final关键字的用法、Synchronized和lock的区别、volatile的介绍、Syncronized锁的作用、构造函数和构造函数重载的概念、方法覆盖和方法重载的区别、反射获取和设置对象私有字段的值的方法、通过反射创建对象的方式以及内部类的详解。 ... [详细]
  • SpringBoot uri统一权限管理的实现方法及步骤详解
    本文详细介绍了SpringBoot中实现uri统一权限管理的方法,包括表结构定义、自动统计URI并自动删除脏数据、程序启动加载等步骤。通过该方法可以提高系统的安全性,实现对系统任意接口的权限拦截验证。 ... [详细]
  • CF:3D City Model(小思维)问题解析和代码实现
    本文通过解析CF:3D City Model问题,介绍了问题的背景和要求,并给出了相应的代码实现。该问题涉及到在一个矩形的网格上建造城市的情景,每个网格单元可以作为建筑的基础,建筑由多个立方体叠加而成。文章详细讲解了问题的解决思路,并给出了相应的代码实现供读者参考。 ... [详细]
  • 开发笔记:实验7的文件读写操作
    本文介绍了使用C++的ofstream和ifstream类进行文件读写操作的方法,包括创建文件、写入文件和读取文件的过程。同时还介绍了如何判断文件是否成功打开和关闭文件的方法。通过本文的学习,读者可以了解如何在C++中进行文件读写操作。 ... [详细]
  • 海马s5近光灯能否直接更换为H7?
    本文主要介绍了海马s5车型的近光灯是否可以直接更换为H7灯泡,并提供了完整的教程下载地址。此外,还详细讲解了DSP功能函数中的数据拷贝、数据填充和浮点数转换为定点数的相关内容。 ... [详细]
  • 模板引擎StringTemplate的使用方法和特点
    本文介绍了模板引擎StringTemplate的使用方法和特点,包括强制Model和View的分离、Lazy-Evaluation、Recursive enable等。同时,还介绍了StringTemplate语法中的属性和普通字符的使用方法,并提供了向模板填充属性的示例代码。 ... [详细]
  • 解决.net项目中未注册“microsoft.ACE.oledb.12.0”提供程序的方法
    在开发.net项目中,通过microsoft.ACE.oledb读取excel文件信息时,报错“未在本地计算机上注册“microsoft.ACE.oledb.12.0”提供程序”。本文提供了解决这个问题的方法,包括错误描述和代码示例。通过注册提供程序和修改连接字符串,可以成功读取excel文件信息。 ... [详细]
  • 本文介绍了使用哈夫曼树实现文件压缩和解压的方法。首先对数据结构课程设计中的代码进行了分析,包括使用时间调用、常量定义和统计文件中各个字符时相关的结构体。然后讨论了哈夫曼树的实现原理和算法。最后介绍了文件压缩和解压的具体步骤,包括字符统计、构建哈夫曼树、生成编码表、编码和解码过程。通过实例演示了文件压缩和解压的效果。本文的内容对于理解哈夫曼树的实现原理和应用具有一定的参考价值。 ... [详细]
  • GreenDAO快速入门
    前言之前在自己做项目的时候,用到了GreenDAO数据库,其实对于数据库辅助工具库从OrmLite,到litePal再到GreenDAO,总是在不停的切换,但是没有真正去了解他们的 ... [详细]
  • dotNet变量和数据类型详解,包括声明变量和五大类型
    本文详细介绍了dotNet编程中的变量和数据类型,包括声明变量和五大类型(int、double、decimal、string、char)。文章通过案例演示了变量的声明和赋值方法,并解释了每种数据类型的特点和使用场景。此外,还介绍了变量命名规则和一些特殊情况,如String与string的区别、float类型的使用等。阅读本文可以帮助读者更好地理解和应用dotNet编程中的变量和数据类型。 ... [详细]
  • 使用eclipse创建一个Java项目的步骤
    本文介绍了使用eclipse创建一个Java项目的步骤,包括启动eclipse、选择New Project命令、在对话框中输入项目名称等。同时还介绍了Java Settings对话框中的一些选项,以及如何修改Java程序的输出目录。 ... [详细]
  • EPPlus绘制刻度线的方法及示例代码
    本文介绍了使用EPPlus绘制刻度线的方法,并提供了示例代码。通过ExcelPackage类和List对象,可以实现在Excel中绘制刻度线的功能。具体的方法和示例代码在文章中进行了详细的介绍和演示。 ... [详细]
  • 微信官方授权及获取OpenId的方法,服务器通过SpringBoot实现
    主要步骤:前端获取到code(wx.login),传入服务器服务器通过参数AppID和AppSecret访问官方接口,获取到OpenId ... [详细]
author-avatar
5jkd_330
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有