本文借鉴《编写高质量的C#代码:改善C#程序的157个建议》,算是对自己学习的总结,也希望分享下所学知识~~
如何操作字符串?
如何进行转型?
什么是克隆?
什么时候需要用HashCode?
这些看似简单的问题都知道吗?
字符串是很频繁使用的集中基础数据类型,使用不当很容易带来额外的性能开销。
case 1:
string str1 = "str1" + 9;
string str2 = "str2" + 9.ToString();
这两行代码看起来一样的代码,生成出来的IL代码大有不同。
(关于如何生成IL代码,有时间在介绍,TODO)
第一行代码,在运行时会完成一次装箱行为(IL中的box),而第二行代码并没有发生装箱,调用的是整形的ToString()方法:
public override String ToString()
{
return Number.FormatInt32( ... )
}
这是一个非托管方法,直接操作内存来完成int到string的转换,效率要比装箱高很多。
装箱之所以会带来性能消耗,会完成三个步骤:
1.为值类型在托管堆中分配内存。除了本身的内存还需要对象指针和同步索引所占用的内存。
2.将值类型的值复制到新分配的堆内存中。
3.返回已经成为引用类型的对象的地址。
case 2:
String是一个很特殊的对象,一旦赋值就不可改变。使用“=”或“+”,都会在内存中创建一个新的字符串对象,也就是分配新的内存空间。
string s1 = "abc";
s1 = "123" + s1 + "456";
string s2 = "123" + "abc" + "456";
上面的两行代码,创建了三个字符串,执行了一次string.Contact()方法。
如果改成最后一行代码,就会在编译时直接生成一个字符串:
string a = "t";
a += "e";
a += "s";
a += "t";
string a = "t";
string b = "e";
string c = "s";
string d = "t";
string result = a + b + c + d;
上面两种方式的区别:
- 同样的创建了四个字符串
- 第一种执行了三次string.Contact(),而第二种只执行了一次
如果要使用字符串拼接,请使用 StringBuilder
,其效率源于预先以非托管的方式分配内存。
但需要注意的是其默认长度为16,一旦字符长度大于16,又会重新分配内存。
ps:string.Format()
内部就是使用了StringBuilder进行的字符串格式化。
case 3:
为类型输出格式化字符串:
class A : IFormattable
{
string str1 = "123";
string str2 = "456";
public string ToString(string format, IFormatProvider formatProvider)
{
switch (format)
{
case "-":
return string.Format("{0}-{1}", str1, str2);
case ".":
return string.Format("{0}.{1}", str1, str2);
default:
return base.ToString();
}
}
}
//for test
A a = new A();
Console.WriteLine(a.ToString());//BetterCode.Program+A
Console.WriteLine(a.ToString("-", null));//123-456
Console.WriteLine(a.ToString(".", null));//123.456