字符串对我编程人员来说是字符串时每天见面的常客,你不认识不熟悉他都不得行,字符串的拼接更是家常便饭,那么在实际开发过程中实现字符串的拼接有哪一些方式呢?咱们一起来聊聊,来交流沟通,学习一波。也许你会说,那也太简单了嘛,谁不会啊,哈哈,使用起来确实简单,但是不一定我们都使用的方式还有优秀的方式吗?

在文章前,我们先简单聊聊关于string的数据类型存储必须了解概念:

string是一个引用类型,是一个sealed类,存储在堆内存上,每一次修改都会从新创建一个新的string来存储,原始的会自动被回收。这个是不感觉是废话,人人都知道嘛,哈哈哈。

下面以c#为开发语言来说明:实现字符串的拼接常用的方式有如下四种

其一、直接通过+拼接

直接通过+拼接是我们在代码中最常见的一种方式,下面以一个简单的代码段来分析分析

     string str="";

     str=str+"";

第一段代码,首先分配了一个内存空间来存储str变量,其值为“1”

第二段代码,重新分配了一个新的内存空间来存储“12”,并将str指向新地址

通过分析,其实我们不难发现,两端就简单的代码,就会有两次内存地址操作,随着拼接字符串的个数地址,分配内存地址的次数也递增,当几个简单的字符串通过该方式拼接时,其实我们还是感觉不到性能的影响,但是当字符串数量大时,你都会有感觉了,那样不仅仅造成内存的浪费,还直接影响性能。

所以在实际开发工程中,通过+拼接字符串比较常见,但是如果只是见到这种方式也就不那么友好了,既然不友好,那么显然就会有比较友好的方式啦,下面我们就分析分析通过StringBuilder来实现字符串的拼接。

其二、通过StringBuilder拼接字符串

 

StringBuilder其实内部相当于是维护的一个字符数组,是一个可以动态增加自身数据长度,其默认长度为16,当存储的字符串超出其长度是,会自动扩容2倍长度。

哈哈,说到这儿,估计你看出了问题,那就是超出长度自动扩容,自动扩容是不是也需要牺牲性能,当然在几次扩容你还感觉不到性能的影响,但是如果词数多了,你就会感觉很明显,这也是对StringBuilder的一些使用技巧。

我们去看不同小伙伴的代码,你就会发现,技术老鸟,在初始化StringBuilder的时候会根据预估将要存储的字符串大小,给StringBuilder初始化一个长度,这也就是细节上的差距体现。

说了半天的废话,是不是要来的实际的代码来证明说的不是废话呢?不急不急,在文章最后,我会专门写测试代码对比分析的。

其三、string.Format不陌生吧

对于一些格式的数据拼接填充,string.Format也是经常看见的,他的一个很大好处就是,看上去比较清晰

其实我们看过string的底层实现我们会发现,其底层本质还是StringBuilder来实现的

下面就是string.format的源码实现

public static String Format(IFormatProvider provider, String format, params Object[] args) <br>{

    if (format == null || args == null)

      throw new ArgumentNullException((format==null)?"format":"args");

    StringBuilder sb = new StringBuilder(format.Length + args.Length * 8);

    sb.AppendFormat(provider,format,args);

    return sb.ToString();

}

其实string.Format使用起来很简单,我就不在啰嗦介绍了,免得大家觉得烦,哈哈哈

string result=string.Format("大家好,我叫{0},今年{1}","程序员修炼之旅",1);

其四、$方式拼接字符串

 

C#6.0出现了$方式拼接字符串,其实简单说就是string.Format简化操作版,string.Format如果拼接的字符串太多,估计自己都懵逼的分不清对应关系了,不知道你们遇到过没有,反正我原来是遇到过的。$就很好的规避了该问题,那么下    面来一个例子说明一切:

string name = "程序员修炼之旅";​
int age = 1;​
string str = string.Format("my name is{0}, I'm {1} years old",name,age);​
string str2 = $"my name is{name}, I'm {age} years old";​
最终结果是:str=str1

其五,当然还有其他方式,不在此啰嗦了,后续在讨论

 

测试分析

说了半天,不拿点实际东西来测试,我知道你是不会信服的,下面就直接上测试代码:

using System;
using System.Diagnostics;
using System.Text; namespace stringSplicingTest
{
/// <summary>
/// 字符串拼接练习
/// </summary>
public class Program
{
/// <summary>
/// 主函数入口
/// </summary>
/// <param name="args"></param>
static void Main(string[] args)
{
// 测试分别通过+ 和 StringBuilder 来连接 0 之100的数字
Console.WriteLine("测试分别通过+ 和 StringBuilder 来连接");
Console.WriteLine("");
Console.WriteLine("测试连接 0 - 100 的数字");
Console.WriteLine("");
PlusString(100);
StringBuilderString2(100);
Console.WriteLine(""); Console.WriteLine("");
Console.WriteLine("测试连接 0 - 10000 的数字");
PlusString(10000);
StringBuilderString2(10000);
Console.WriteLine("");
Console.WriteLine(""); // 下面测试一下同样是StringBuilder连接字符串,一个是定义吃指定长度,一个是不指定长度对比
Console.WriteLine(@"下面测试一下同样是StringBuilder连接字符串, 一个是定义并指定长度,一个是不指定长度对比");
Console.WriteLine("");
Console.WriteLine("测试连接 0 - 1000000 的数字");
Console.WriteLine("不初始化长度");
StringBuilderString(1000000);
Console.WriteLine("初始化长度");
StringBuilderString2(1000000); Console.WriteLine("");
Console.WriteLine(""); Console.WriteLine("测试连接 0 - 10000000 的数字");
Console.WriteLine("不初始化长度");
StringBuilderString(10000000);
Console.WriteLine("初始化长度");
StringBuilderString2(10000000); Console.ReadLine();
} /// <summary>
/// 通过+拼接字符串
/// </summary>
/// <param name="totalNum"></param>
private static void PlusString(int totalNum)
{
//// 定义一个秒表,执行获取执行时间
Stopwatch st = new Stopwatch();//实例化类
st.Start();//开始计时 Console.WriteLine("开始执行,通过+连接字符串:");
string result = "";
//// 定义一个数组
for (int i = 0; i < totalNum; i++)
{
result = result + i.ToString();
} //需要统计时间的代码段 st.Stop();//终止计时
Console.WriteLine(string.Format("执行完毕,通过+连接字符串!总耗时{0}毫秒",
st.ElapsedMilliseconds.ToString()));
} /// <summary>
/// 通过s拼接字符串
/// </summary>
/// <param name="totalNum"></param>
private static void StringBuilderString(int totalNum)
{
//// 定义一个秒表,执行获取执行时间
Stopwatch st = new Stopwatch();//实例化类
st.Start();//开始计时 Console.WriteLine("开始执行,通过 StringBuilder 连接字符串:"); StringBuilder result = new StringBuilder(); //// 定义一个数组
for (int i = 0; i < totalNum; i++)
{
result.Append(i.ToString());
} string result2 = result.ToString();
//需要统计时间的代码段 st.Stop();//终止计时
Console.WriteLine(string.Format("执行完毕,通过 StringBuilder 连接字符串!总耗时{0}毫秒",
st.ElapsedMilliseconds.ToString()));
} /// <summary>
/// 通过StringBuilder拼接字符串,初始化时指定一个长度
/// </summary>
/// <param name="totalNum"></param>
private static void StringBuilderString2(int totalNum)
{
//// 定义一个秒表,执行获取执行时间
Stopwatch st = new Stopwatch();//实例化类
st.Start();//开始计时 Console.WriteLine("开始执行,通过 StringBuilder 连接字符串:"); StringBuilder result = new StringBuilder(totalNum * 6); //// 定义一个数组
for (int i = 0; i < totalNum; i++)
{
result.Append(i.ToString());
} string result2 = result.ToString();
//需要统计时间的代码段 st.Stop();//终止计时
Console.WriteLine(string.Format("执行完毕,通过 StringBuilder 连接字符串!总耗时{0}毫秒",
st.ElapsedMilliseconds.ToString()));
}
}
}

结果分析总结:

测试分两个点:

其一测试的是:通过+和StringBuilder拼接字符串的性能比较哦

其二测试的是:StringBuilder初始化长度和不初始化长度的性能比较

大概得出以下几点结论

1、在待拼接的字符串少的时,+和StringBuilder没有明显的性能差距

2、当拼接的字符串多时,StringBuilder的优势越来越明显

3、同样是StringBuilder拼接字符串,预估初始化长度的效率比不初始化指定长度的效率高

说到此,我相信大家都知道该怎么使用了。好了,时间不早了,赶紧洗洗睡了,明天还得上班呢?

END

 原文地址:https://www.cnblogs.com/xiaoXuZhi/p/XYH_String2.html

聊聊c#字符串拼接的更多相关文章

  1. 从源代码的角度聊聊java中StringBuffer、StringBuilder、String中的字符串拼接

    长久以来,我们被教导字符串的连接最好用StringBuffer.StringBuilder,但是我们却不知道这两者之间的区别.跟字符串相关的一些方法中总是有CharSequence.StringBuf ...

  2. SQL中字符串拼接

    1. 概述 在SQL语句中经常需要进行字符串拼接,以sqlserver,oracle,mysql三种数据库为例,因为这三种数据库具有代表性. sqlserver: select '123'+'456' ...

  3. StringBuilder(字符串拼接类)

    StringBuilder是在using System.Text命名空间下的一个成员. 在做字符串拼接的时候,因为字符串是引用类型,新的字符串是会再内存中创建的,所以用+号拼接字符串是比较耗效率的. ...

  4. 精简版StringBuilder,提速字符串拼接

    编写目的 在频繁的字符串拼接中,为了提升程序的性能,我们往往会用StringBuilder代替String+=String这样的操作; 而我在实际编码中发现,大部分情况下我用到的只是StringBui ...

  5. Objective的字符串拼接 似乎没有Swift方便,但也可以制做一些较为方便的写法

    NSString *str1 = @"字符串1"; NSString *str2 = @"字符串2"; //在同样条件下,Objective的字符串拼接 往往只 ...

  6. java 创建string对象机制 字符串缓冲池 字符串拼接机制

    对于创建String对象的机制,在这一过程中涉及的东西还是值得探究一番的. 首先看通过new String对象和直接赋值的方式有什么区别,看如下代码: public static void main( ...

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

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

  8. 【SQL】小心字符串拼接导致长度爆表

    请看代码: DECLARE @max VARCHAR(max) SET @max='aaa...' --这里有8000个a +'bb' --连接一个varchar常量或变量 SELECT LEN(@m ...

  9. WMSYS.WM_CONCAT(distinct(字段名)) 函数,字符串拼接函数。合并列

    合并列函数 WMSYS.WM_CONCAT(distinct(字段名)) 函数 可以实现字符串拼接在一起,这种情况可以在要求把一个字段的多个值拼接在一起的时候使用.其中distinct可以去掉重复的值 ...

随机推荐

  1. Android Studio 3.6 正式版终于发布了

    Google 下载地址 百度云 下载地址 密码:epl9 如题,Android Studio 3.6 正式版终于发布了,值得兴奋呀,毕竟 3.5 大版本更新也已经差不多半年了,撒花撒花!这次更新又更新 ...

  2. 【二】、UML基础知识——图图解乾坤

    [二].UML基础知识 UML概述 UML是一个通用的可视化建模语言,不同于编程语言,它通过一些标准的图形符号和文字来对系统进行建模.用于对软件进行描述.可视化处理.构建软件系统的文档.是一套总结了以 ...

  3. Java 添加超链接到Excel文档

    超链接即内容链接,通过给特定对象设置超链接,可实现载体与特定网页.文件.邮件.网络等的链接,点击链接载体可打开链接目标,在文档处理中是一种比较常用的功能.本文将介绍通过Java程序给Excel文档添加 ...

  4. codewars--js--Hamming Numbers

    问题描述: A Hamming number is a positive integer of the form 2i3j5k, for some non-negative integers i, j ...

  5. 你一定看得懂的 DDD+CQRS+EDA+ES 核心思想与极简可运行代码示例

    前言 随着分布式架构微服务的兴起,DDD(领域驱动设计).CQRS(命令查询职责分离).EDA(事件驱动架构).ES(事件溯源)等概念也一并成为时下的火热概念,我也在早些时候阅读了一些大佬的分析文,学 ...

  6. 复杂系统架构设计<1>

    这两天开始读由Edward Crawley(爱德华 克劳利).Bruce Cameron(布鲁斯 卡梅隆).Daniel Selva(丹尼尔 塞尔瓦)著作的系统架构,一开始看目录以为是介绍系统软件架构 ...

  7. 史上最全的Java高级技术点,全是Java高级进阶技术,几乎包含了Java后端的所有知识点

    史上最全的Java高级技术点,全是Java高级进阶技术,几乎包含了Java后端的所有知识点 1

  8. Python之write与writelines区别

    一.传入的参数类型要求不同: 1. file.write(str)需要传入一个字符串做为参数,否则会报错. write( "字符串") with open('20200222.tx ...

  9. springboot + mybatis 支持oracle和mysql切换含源码

    1.springboot 启动类加入bean 如下 // DatabaseIdProvider元素主要是为了支持不同的数据库@Beanpublic DatabaseIdProvider getData ...

  10. 2.Python函数/方法(method/function)详解

    1.什么是函数 它是一段功能代码,理解为一种功能行为,在内存中有空间区域,函数需要被调用才能执行(通过函数名来调用): 好处: 1).提高代码的复用性 2).提升代码的阅读性 3).增加代码的扩展性 ...