C_中字符串操作总结_第1页
C_中字符串操作总结_第2页
C_中字符串操作总结_第3页
C_中字符串操作总结_第4页
C_中字符串操作总结_第5页
已阅读5页,还剩13页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

1、C#中字符串操作总结在实际的开发过程中,对字符串的操作是经常遇到的,其中涉及到字符串拼接、拆分、比较、替换等操作。C#提供了string 类型,String和StringBuilder 两种类来对字符串进行处理。那么string,String,StringBuilder对字符串进行处理有何异同,在实际编程中,对于不同的字符串操作应该采用哪种方式来提高程序的效率呢?本文将对string,String,StringBuilder进行详细的解释和比较,最后在编程过程中遇到的常用的字符串处理进行了总结。首先理解string,String,StringBuilder的概念和区别:stringstring

2、,msdn 给出的解释就是,string 是C#中的关键字,并且是引用类型,string类型表示零或更多Unicode 字符组成的序列。string 是.NET Framework 中String 的别名。但定义相等运算符(=和!=是为了比较string 对象(而不是引用的值(后面给出示例解释这点。String:String 是类,表示文本,即一系列Unicode 字符。String对象是不可改变的。每次使用System.String 类中的方法之一时,都要在内存中创建一个新的字符串对象,这就需要为该新对象分配新的空间。如:当我们实例化一个String 的对象后,在内存中为此对象分配一个空间。

3、如下:Stringstr =“hello”;当我们修改str 的值的时候,如:str=“helloworld”;此时,系统会为str 重新分配一个空间。这样原来的内存空间就被浪费掉了,只能等待垃圾回收器回收。在需要对字符串执行重复修改的情况下,与创建新的String 对象相关的系统开销可能会非常昂贵。String 与string 的区别:string 是.NET Framework 中String 的别名,string是C#基元类型(primitive,简单来说就是编译器直接支持的数据类型。基元类型要直接映射到Framework 类库(FCL中的类型,例如,C#中一个基元类型int 直接映射到

4、System.Int32类型,这里int 是基元类型,System.Int32是FCL 类型。而String 是FCL 类型的,所以在C#的编译时,会自动的把string 转化为System.String。所以string 与String 实质上没什么区别,只是在使用string 要做一次转换,转换为String。因此,在编码时我们推荐使用String。string 虽然为引用类型,但是(=和!=是为了比较string 对象(而不是引用的值。如:string a ="hello" ;string b ="hel" ;b =b +"lo"

5、; ;string c ="hello" ;Console.WriteLine(a=b;/TrueConsole.WriteLine(object a =(object b; /FalseConsole.WriteLine(object a =(object c; /Truea=b比较的是值而非引用。所以a=b为True。当创建多个字符串内容相同的对象时,都只会指向同一个引用;a 和c 都指向同一个a 的引用,并不会为c 重新分配内存;这样即可保证内存有效利用;所以上面的(objecta=(objectc比较的是a 与c 的引用,结果为True。这里面b 由于进行了累加操作

6、(b=b +"lo"b 又重新分配了内存,所以(objecta=(objectb比较的是引用,所以为False。接下来我们再通过几个例子来理解下String(stringString str1="abc" ;String str2=str1; /使str2和str1指向同一引用,即指向同一内存地址str1="123" ; /a跑掉原来的对象而生成一个新的字符串对象,分配新的内存空间。Console.WriteLine(str2;/abc输出结果是abc,首先给str 赋值为"abc",接着执行str2=str1,使

7、str2和str1指向同一引用,即内存地址。当执行str1="123"后,String对象是不可改变的,实质上str1="123"是str1=newstring("123"的简写,它的每一次赋值都会抛掉原来的对象而生成一个新的字符串对象,分配新的内存空间,str1="123"语句编译器私底下创建了一个新的字符串对象来保存新的字符序列"123",也就是此str1已非彼str1了。因此str1的值的改变也就不能影响先前str1指向地址的值了,当然str2的值也就不会改变了。因此string 是不可改

8、变的。通过上面的例子,如果我们执行下面这些语句:String sql =“Select*From T_Test”;sql +=“Whereid=888”;sql +=“Andtype=3”;sql +=“OrderBy Desc”;实际上这样是十分浪费内存空间的。如果是频繁的这样做的话,建议是使用StringBuilder 对象,或者这样写:String sql =“Select*From T_Test”+“Whereid=888”+“Andtype=3”+“OrderBy Desc ”;StringBuilder:出于性能方面的考虑,大量的串联或所涉及其他字符串操作应通过StringBuil

9、der 类来执行。StringBuilder表示可变字符字符串,它允许我们有效的对字符串的字符执行动态操作,有效的缩减字符串的大小或者更改字符串中的字符。如果字符串变大,超过已经分配的字符的大小,StringBuilder就会自动的分配一个全新的、更大的数组,并开始使用新的数组,虽然StringBuilder 对象是动态对象,允许扩充它所封装的字符串中字符的数量,但是您可以为它可容纳的最大字符数指定一个值。此值称为该对象的容量,不应将它与当前StringBuilder 对象容纳的字符串长度混淆在一起。例如,可以创建StringBuilder 类的带有字符串“Hello”(长度为5的一个新实例,

10、同时可以指定该对象的最大容量为25。当修改StringBuilder 时,在达到容量之前,它不会为其自己重新分配空间。当达到容量时,将自动分配新的空间且容量翻倍。可以使用重载的构造函数之一来指定StringBuilder 类的容量。String 或StringBuilder 对象的串联操作的性能取决于内存分配的发生频率。String串联操作每次都分配内存,而StringBuilder 串联操作仅当StringBuilder 对象缓冲区太小而无法容纳新数据时才分配内存。因此,如果串联固定数量的String 对象,则String 类更适合串联操作。这种情况下,编译器甚至会将各个串联操作组合到一个操

11、作中。如果串联任意数量的字符串,则StringBuilder 对象更适合串联操作;例如,某个循环对用户输入的任意数量的字符串进行串联。C#中判断空字符串首先明确”,null和string.Empty 的区别:string.Empty:不分配空间。"":分配一个长度为空的存储空间,""和String.Empty,这两个都是表示空字符串,空字符串是一个特殊的字符串,只不过这个字符串的值为空,在内存中是有准确的指向的。string.Empty 就相当于"",一般用于字符串的初始化。比如:string a =string.Empty;在进行为

12、空的比较时。string.Empty和""是一样的。即如果string test1=""则可以使用if(test1=""或者if(test1=string.Empty进行判断。上面两句是一样的效果。Null:null关键字是表示不引用任何对象的空引用的文字值。null是引用类型变量的默认值。那么也只有引用型的变量可以为NULL,如果int i=null,的话,是不可以的,因为Int 是值类型的。String.Empty 和Null,这两个都是表示空字符串,stringstr1=String.Empty,这样定义后,str1是一个空字符

13、串,空字符串是一个特殊的字符串,只不过这个字符串的值为空,在内存中是有准确的指向的,stringstr2=null,这样定义后,只是定义了一个string 类的引用,str2并没有指向任何地方,在使用前如果不实例化的话,都将报错。所以下面代码中执行test3.Length =0就是错误的。判断空字符串:string test1="" ;string test2=string .Empty;string test3=null ;Console.WriteLine("test1="""+" " ;Console.Wri

14、teLine("test2=string.Empty" "</br>"Console.WriteLine("test3=null" +"</br>"if (test1="" Console.WriteLine("(test1=""is :True"+"</br>"if (test2=string .EmptyConsole.WriteLine("(test2=string.Empty is:

15、True"+"</br>"if (test1=string .EmptyConsole.WriteLine("(test1=string.Empty is:True" +"</br>"if (test2="" Console.WriteLine("(test2=""is:True" +"</br>"if (test1=test2Console.WriteLine("(test1=test2 is:Tr

16、ue" +"</br>"if (test3=null Console.WriteLine("(test3=null is:True" +"</br>"if (test1!=null Console.WriteLine("(test1!=null is :True" +"</br>"if (test2!=null Console.WriteLine("(test2!=null is :True" +"</br>

17、"if (test1.Length=0Console.WriteLine("(test1.Length=0is:True" +"</br>"if (test2.Length=0Console.WriteLine("(test2.Length=0is :True" +"</br>"/if(test3.Length=0/Error,null不能用Length 来进行判断为空if (string .IsNullOrEmpty(test1Console.WriteLine("(s

18、tring.IsNullOrEmpty(test1is :True"+"</br>"if (string .IsNullOrEmpty(test2Console.WriteLine("(string.IsNullOrEmpty(test2is :True"+"</br>"if (string .IsNullOrEmpty(test3Console.WriteLine("(string.IsNullOrEmpty(test3is :True"+"</br>&qu

19、ot;输出:test1=""test2=string .Emptytest3=null(test1="" is :True(test2=string .Empty is :True(test1=string .Empty is :True(test2="" is :True(test1=test2 is :True(test3=null is :True(test1!=null is :True(test2!=null is :True(test1.Length=0 is :True(test2.Length=0 is :True(s

20、tring .IsNullOrEmpty(test1is :True(string .IsNullOrEmpty(test2is :True(string .IsNullOrEmpty(test3is :True因此,判断字符串为空最通用的方法就是IsNullOrEmpty(无论是"",string.Empty 还是null。如果字符串初始化为null,则不能使用test3.Length =0进行判断。对于"",和string.Empty 使用s.Length =0,s=string.Empty 和s =""都可以,这里面不讨论性能问

21、题。比较字符串相等Equals(和运算符=C#中有两种不同的相等:引用相等和值相等。值相等是大家普遍理解的意义上的相等:它意味着两个对象包含相同的值。例如,两个值为2的整数具有值相等性。引用相等意味着要比较的不是两个对象,而是两个对象引用,且两者引用的是同一个对象。这可以通过简单的赋值来实现,如下面的示例所示:System.Object a =new System.Object(;System.Object b =a;System.Object.ReferenceEquals(a,b; /returnstrue在上面的代码中,只存在一个对象,但存在对该对象的多个引用:a和b。由于它们引用的是同

22、一个对象,因此具有引用相等性。如果两个对象具有引用相等性,则它们也具有值相等性,但是值相等性不能保证引用相等性。若要检查引用相等性,应使用ReferenceEquals。若要检查值相等性,请使用Equals。运算符=对于预定义的值类型,如果操作数的值相等,则相等运算符(=返回true,否则返回false。对于string 以外的引用类型,如果两个操作数引用同一个对象,则=返回true。对于string 类型,=比较字符串的值对于内置值类型,=判断的是两个对象的代数值是否相等。它会根据需要自动进行必要的类型转换,并根据两个对象的值是否相等返回true 或者false。例如:int x =100;

23、Double y =100;if (x=yConsole.WriteLine("x=y!"/x=y而对于用户定义的值类型,如果没有重载=操作符,=将是不能够使用的。例如:Struct Userstruct1;Userstruct1a;Userstruct1b;If(a=bConsole.WriteLine(“can=reach this far?”上面的这段代码是不能够通过编译的。可以通过重载使=作用于用户定义的值类型。对于引用类型,=默认的行为与ReferenceEquals 的行为相同,仅有两个对象指向同一个Reference 的时候才返回true。Equals(:Eq

24、uals 方法对于值类型和引用类型的定义不同,对于值类型,类型相同,并且数值相同(对于struct 的每个成员都必须相同,则Equals 返回true,否则返回false。而对于引用类型,默认的行为与ReferenceEquals 的行为相同,仅有两个对象指向同一个Reference 的时候才返回true。可以根据需要对Equals 进行重载,例如String 类的Equals 用于判断两个字符串的内容是否相等。StringBuilder a =new StringBuilder(;a.Append("thetest a"String s1=a.ToString(;Stri

25、ng s2="the test a"if (s2=s1 /为真,因为S2,S1内容相等。String类的=与Equals 的行为相同,判断两个字符串的内容是否相等Console.WriteLine("=returns true"if (Object.Equals(s2,s1 /为真,String类重载了Equals,用于判断两个字符串的内容是否相等Console.WriteLine("equalsreturns true"if (Object.ReferenceEquals(s2,s1 /为假,因为s2,s1并不是指向同一个对象Con

26、sole.WriteLine("ReferenceEqualsreturns true"注:对于String 类,直接声明s1=“thetest a”的话,输出结果将包含"ReferenceEquals returns true",因为默认的,String对于声明的相同的字符串在堆上只保留一个Copy,所以s1与s2将会指向相同的Reference。分割字符串Split 方法将字符串分隔后返回字符串数组。Split 使用方法一:string str ="abcdefghigkceofuecto"stringsArray =str.Sp

27、lit('c'foreach (stringi in sArrayConsole.WriteLine(i.ToString(+"</br>"输出下面的结果:abdefghigkeofuetoSplit 使用方法二:我们看到了结果是以一个指定的字符进行的分割。使用另一种构造方法对多个字符进行分割:string str ="abcdefghigkceofuecto"stringsArray =str.Split(newchar3'h', 'i', 'g' ;foreach (stri

28、ngi in sArrayConsole.WriteLine(i.ToString(+"</br>"输出:abcdefkceofuecto使用正则表达式:使用正则表达式,可以过滤掉一些不是很规则的字符。string str ="everybody*Welcome*to*china*!"stringsArray =System.Text.RegularExpressions.Regex.Split(str,"*+"foreach (stringi in sArrayConsole.WriteLine(i.ToString(+

29、"</br>"输出:everybodyWelcometochina!字符串截取string myString ="Hello Word!"/Substring(在C#中有两个重载函数/分别如下示例string subString1=myString.Substring(0;/如果传入参数为一个长整,且大于等于0,/则以这个长整的位置为起始,/截取之后余下所有作为字串./如若传入值小于0,/系统会抛出ArgumentOutOfRange 异常/表明参数范围出界string subString2=myString.Substring(0,5;/如

30、果传入了两个长整参数,/前一个为参数子串在原串的起始位置/后一个参数为子串的长度/如不合条件同样出现上述异常Console.WriteLine(subString1;/HelloWord!Console.WriteLine(subString2;/Hello格式化字符串格式化数字格式字符说明和关联属性_c、C货币格式。d、D十进制格式。e、E科学计数(指数格式。f、F固定点格式。g、G常规格式。n、N数字格式。P、P百分比r、R往返格式,确保将已转换成字符串的数字转换回数字时具有与原数字相同的值。x、X十六进制格式。_double val=Math.PI;Console.WriteLine(v

31、al.ToString(; /displays3.14159265358979Console.WriteLine(val.ToString("E"/displays3.141593E+000Console.WriteLine(val.ToString("F3"/displays3.142int val=65535;Console.WriteLine(val.ToString("x"/displaysffffConsole.WriteLine(val.ToString("X"/displaysFFFFSingle v

32、al=0.123F;Console.WriteLine(val.ToString("p"/displays12.30%Console.WriteLine(val.ToString("p1"/displays12.3%格式化日期d 短日期模式表示由当前ShortDatePattern 属性定义的自定义DateTime 格式字符串。例如,用于固定区域性的自定义格式字符串为“MM/dd/yyyy”。G 常规日期/时间模式(短时间表示短日期(d和短时间(t模式的组合,由空格分隔。G 常规日期/时间模式(长时间表示短日期(d和长时间(T模式的组合,由空格分隔。M

33、或m 月日模式表示由当前MonthDayPattern 属性定义的自定义DateTime 格式字符串。例如,用于固定区域性的自定义格式字符串为“MMMMdd”。R 或r RFC1123模式表示由当前RFC1123Pattern 属性定义的自定义DateTime 格式字符串。该模式是定义的标准,并且属性是只读的。因此,无论所使用的区域性或所提供的格式提供程序是什么,它总是相同的。定义格式字符串为“ddd,dd MMM yyyy HH':'mm':'ss'GMT'”。格式化不会修改正在格式化的DateTime 对象的值。因此,应用程序在使用此格式说明

34、符之前必须将该值转换为协调世界时(UTC。T 长时间模式表示由当前LongTimePattern 属性定义的自定义DateTime 格式字符串。例如,用于固定区域性的自定义格式字符串为“HH:mm:ss”。U 通用的可排序日期/时间模式表示由当前UniversalSortableDateTimePattern 属性定义的自定义DateTime 格式字符串。此模式是定义的标准,并且属性是只读的。因此,无论所使用的区域性或所提供的格式提供程序是什么,它总是相同的。自定义格式字符串为“yyyy'-'MM'-'ddHH':'mm':'ss

35、'Z'”。格式化日期和时间时不进行时区转换。因此,应用程序在使用此格式说明符之前必须将本地日期和时间转换为协调世界时(UTC。DateTime dt =DateTime.Now;String date;date =dt.ToString("d" ,DateTimeFormatInfo.InvariantInfo;Console.WriteLine(date+"</br>" /07/22/2009date =dt.ToString("G" , DateTimeFormatInfo.InvariantInfo;

36、Console.WriteLine(date+"</br>" /07/22/200914:54:37date =dt.ToString("r" , DateTimeFormatInfo.InvariantInfo;Console.WriteLine(date+"</br>" /Wed,22Jul 200914:54:37GMTdate =dt.ToString("T" , DateTimeFormatInfo.InvariantInfo;Console.WriteLine(date+&quo

37、t;</br>" /14:54:37date =dt.ToString("u" , DateTimeFormatInfo.InvariantInfo;Console.WriteLine(date+"</br>" /2009-07-2214:54:37Zdate =dt.ToString("dd-MM-yyyy" , DateTimeFormatInfo.InvariantInfo;Console.WriteLine(date+"</br>" /22-07-2009其他函数Endswith 末尾是否匹配指定stringIndexof 索引指向int start 开始的第一个stringInsert 插入stringLength 长度,数组为大小Remove 从string 中删除,数组为删除一个stringReplace 替换 StartsWith 开始是否与指定 string 匹

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论