最近闲来之余,看了一些开源的类库,看到有些类库喜欢用TextWriter类来记录相关的字符串数据,感到比较好奇,为啥不用StringBuilder类对象。于是在网上搜索了一番,总结了相关笔记。

StringBuilder

在 .net 中,字符串作为一种基本的数据类型,通常在一个程序中同一个字符串只维护一个副本。也就是说,通过直接给定字符串值的字符串引用会引用到相同数据上。这种处理的好处在于它能够减少字符串所占用的内存空间,不需要为多个同样的字符串开辟多次空间。在C#中 string 类型是一个不变量,给字符串引用赋予新值并不会改变对应内存中的数据,而是设置引用为新字符串位置。

在平时,这种处理逻辑能够大大减少字符串所占用的内存空间,但有的时候,也会起一些反效果,典型的例子就是在一些构造字符串的操作时所生成的中间字符串数据。举个例子:

string[] words = {"Nice ", "to ", "meet ", "you."};
string sentence = "";
for(i = 0;i < words.Length; i++)
{
sentence += words[i];
}

这是一个很简单的字符串组装功能,它将给定的单词拼接成一个句子,我们希望的是直接拼接成最后的结果,但这段代码除了生成最终句子外,先前的临时也会生成出来。也就是说,"Nice "、"Nice to "、"Nice to meet "以及最后字符串"Nice to meet you."会随着一次次循环迭代全部构造出来。但实际上,对于我们来说只需要最后句子即可,中间部分完全不需要。为此,我们需要新的方式来避免无意义的开销。

StringBuilder类就是一种动态灵活地构造字符串的方法。这种构造字符串的好处在于,它能够避免构造中间字符串结果,转而直接生成最终的字符串数据。按照上面的例子,稍作修改就能得到一个性能更加优异的版本,在该版本下只有最后的句子字符串才会被生成。

string[] words = {"Nice ", "to ", "meet ", "you."};
StringBuilder sentenceBuilder = new StringBuilder();
for(i = 0;i < words.Length; i++)
{
sentenceBuilder.Append(words[i]);
}
string setence = sentenceBuilder.ToString();

至于StringBuilder类的原理,我个人猜测是该类中维护一个char型列表,然后动态地修改数组元素,达到每次拼接时不会生成字符串的目的,只有当显式调用命令生成时,才会生成。不过,因为能力有限,我还不知道怎么在runtime这个开源库中找StringBuilder的实现。

TextWriter

TextWriter是一个抽象类,按照微软官方给出的描述,该类指的是可以编写一个有序字符系列的编写器。嗯,字都认识,但是这句话感觉就不像是人说的。实际上,我个人对这个类的理解是它是一个写入器。换句话来说,TextWriter描述了一个写入的过程,但具体写什么?向哪里写入?这不是这个抽象类所关心的话题,而是由其子类所负责。

.net中常用内置的一些子类:

  • StreamWriter :这个类相信很多人都熟悉,当需要向文件中写入数据时,往往通过该类写入数据
  • StringWriter :今天本文所需要研究的对象,向字符串写入
  • HttpWriter : 向网络流中写入数据

StringWriter类作为TextWriter的一个继承类,按照MSDN给出的解释是,用于将信息写入字符串的TextWriter类对象,这个类看起来和StringBuilder类所做的功能差不多,那么为什么在 .net 中设计两个不同类做同一个功能呢?翻了下相关资料,只能说这两个类是不同设计思路下的产物。StringBuilder是一种灵活构建字符串的类,它不会产生额外的临时字符串,而StringWriter则将字符串数据作为一种写入的目的地,从这个角度来看,确实也是一种必要的实现。

比如说,有一个函数,它专门是将字符串数据记录下来,具体点,可以想像为日志记录器将日志信息记录到某个地方。这样的情况下,我们提供两个输入参数,TextWriter类对象表明是一个写入器,message描述一个日志信息,那么记录数据只需要这样写就可以了:

public void WriteData(TextWriter writer, string message)
{
writer.Write(message);
}

这样一来,如果将信息记录到某个文件中,只要这样写:

using var file = new FileStream("./log.txt", FileAccess.Write);
using var writer = new StreamWriter(file);
WriteData(writer, "hello");

如果想将信息记录到某个变量中,就是这样:

var writer = new StringBuilder();
WriteData(writer, "hello");
data = writer.ToString();

总结

总的来说,如果只是单纯使用字符串而不涉及到修改字符串值时,直接使用string类型即可。如果需求是更加专注构造某种字符串数据,那么使用StringBuilder是一个比较好的选择。如果需求强调的是将某种格式的字符串数据写入到某个介质中,使用TextWriter对应的继承类会更好,更符合封装的思想,且不需要过多关注数据是怎么写入的,只要将需要写入的数据传入到其中即可。

.net 中的 StringBuilder 和 TextWriter 区别的更多相关文章

  1. java中string stringbuilder stringbuffer 的区别

    1. String 类 String的值是不可变的,这就导致每次对String的操作都会生成新的String对象,不仅效率低下,而且大量浪费有限的内存空间. String a = "a&qu ...

  2. JAVA之旅(十七)——StringBuffer的概述,存储,删除,获取,修改,反转,将缓存区的数据存储到数组中,StringBuilder

    JAVA之旅(十七)--StringBuffer的概述,存储,删除,获取,修改,反转,将缓存区的数据存储到数组中,StringBuilder 讲完String,我们来聊聊他的小兄弟 一.StringB ...

  3. final,finally,finalize有什么区别?String, StringBuffer, StringBuilder有什么区别?Exception和Error有什么区别?

    继上篇JVM学习之后,后面将分三期深入介绍剩余JAVA基础面试题,每期3题. 题目一.final,finally,finalize有什么区别? /*请尊重作者劳动成果,转载请标明原文链接:*/ /* ...

  4. Java中String, StringBuilder和StringBuffer

    Java中常用来处理字符串的类有三个: String, StringBuffer和StringBuilder. 区别 三者都继承自CharSequence接口, 首先说明三者间主要区别 String字 ...

  5. JAVA基础之——String、StringBuilder、StringBuffer区别和使用场景

    本文主要讲解String.StringBuilder.StringBuffer区别和应用场景 本文以jdk1.8为例 1 String 操作过程:每次赋值时新建一个String对象. 2 String ...

  6. 浅谈JS中的!=、== 、!==、===的用法和区别 JS中Null与Undefined的区别 读取XML文件 获取路径的方式 C#中Cookie,Session,Application的用法与区别? c#反射 抽象工厂

    浅谈JS中的!=.== .!==.===的用法和区别   var num = 1;     var str = '1';     var test = 1;     test == num  //tr ...

  7. 简单的理解 StringBuffer/StringBuilder/String 的区别

    StringBuffer/StringBuilder/String 的区别 这个三类之间主要的区别:运行速度,线程安全两个方面. 速度方面(快到慢): StringBuilder > Strin ...

  8. 【转】为什么我们都理解错了HTTP中GET与POST的区别

    GET和POST是HTTP请求的两种基本方法,要说它们的区别,接触过WEB开发的人都能说出一二. 最直观的区别就是GET把参数包含在URL中,POST通过request body传递参数. 你可能自己 ...

  9. Visual Studio 中 Build 和 Rebuild 的区别

    因为之前写的程序比较小,编译起来比较快,所以一直都没有太在意 Build 和 Rebuild 之间的区别,后来发现两个还是有很大不同. Build 只针对在上次编译之后更改过的文件进行编译,在项目比较 ...

随机推荐

  1. centos7下安装docker与镜像加速

    1.背景 centos7下安装docker 2.安装 第一步:检查是否为centos7版本 第二步:依赖环境安装 执行如下两个命令: yum -y install gcc yum -y install ...

  2. 基于Prometheus和Grafana打造业务监控看板

    前言 业务监控对许许多多的场景都是十分有意义,业务监控看板可以让我们比较直观的看到当前业务的实时情况,然后运营人员可以根据这些情况及时对业务进行调整操作,避免业务出现大问题. 老黄曾经遇到过一次比较尴 ...

  3. 8.oracle 表查询

    演示如何使用select语句,接下来对emp.dept.salgrade表结构进行解说. emp 雇员表 字段名称 数据类型 是否为空 备注 -------- ----------- -------- ...

  4. 安国AU6989主控 + K9GBG08U0A(NAND) 制作4GB闪存驱动器

    文档标识符:AU6989_FLASH-DRIVE_D-P8 作者:DLHC 最后修改日期:2020.8.22 本文链接: https://www.cnblogs.com/DLHC-TECH/p/AU6 ...

  5. Adversarial Attack Type I: Cheat Classifiers by Significant Changes

    出于实现目的,翻译原文(侵删) Published in: IEEE Transactions on Pattern Analysis and Machine Intelligence (TPAMI ...

  6. Deep Learning-Based Video Coding: A Review and A Case Study

    郑重声明:原文参见标题,如有侵权,请联系作者,将会撤销发布! 1.Abstract: 本文主要介绍的是2015年以来关于深度图像/视频编码的代表性工作,主要可以分为两类:深度编码方案以及基于传统编码方 ...

  7. SPSSAU新功能上线:高级公式、综合得分一键计算!

    一直关注我们的朋友们一定会发现,近期SPSSAU增添了很多新功能. 我们精挑细选出6个最常使用的功能,介绍给大家,看看这些新功能你有没有解锁成功呢? 01 一键删除无效样本 “无效样本”功能中,添加了 ...

  8. 牛客网PAT练习场-数素数

    题解:就是一道素数筛选法 题目地址:https://www.nowcoder.com/questionTerminal/e0fb49acb75f47e8b6fa2077d9071799 /** * C ...

  9. fprintfAndFscanf简单操作

    C语言fscanf和fprintf函数的用法详解(格式化读写文件)(代码转载) #include<stdio.h> #define N 2 struct stu{ char name[10 ...

  10. 断言函数-RF

    测试用例的目的是要验证一些操作否符合我们的预期结果,所以在测试用例中,断言函数是必不可少的一项.我们做的每一步操作都会有预期的结果,为了保证操作得到的结果符合预期,我们需要在测试用例中添加断言,来保证 ...