C#中string.Empty和null的区别详解
这是⼀个及其常见的问题,⽹上已经有关于这个问题的很多讨论。但是我觉得都是不求甚解,有⼀些还是在误导别⼈。下⾯我来说下我对这三者的理解,如有错误的地⽅请⼤家及时指正。
⼀:""与string.Empty我认为是⼀样的。⽹上有⼀篇被转载了⼏⼗遍的⽂章是这样说的string.Empty 不分配存储空间,"" 分配⼀个长度为空的存储空间,我认为这句话是错误并且含糊不清的。
1、实际上Empty是string类中的⼀个静态的只读字段,他的定义是这样的:
public static readonly String Empty = "";
也就是说string.Empty的内部实现是等于””的。
2、我要反驳string.Empty 不分配存储空间,"" 分配⼀个长度为空的存储空间这个观点。⾸先string.Empty与""都会分配存储空间,具体的说是都会在内存的栈和堆上分配存储空间。
有⼀点先说明⼀下,引⽤类型是将对象是实际数据保存在堆上, 将对象在堆上的地址保存在栈上。因此string.Empty与””都会在栈上保存⼀个地址这个地址占4字节,指向内存堆中的某个长度为0的空间,这个空间保存的是string.Empty的实际值。这个我可以⽤VS2010跟踪下内存给⼤家演⽰。
上图中的0x01e81228即是变量str在栈中存储的地址。
对于"",请看下图
这个图的效果跟上图是⼀样的,也就是说””也是在栈上保存了⼀个地址。
3、CLR会对字符串进⾏优化,所以””和string.Empty也都会被优化。
声明如下两个变量
复制代码代码如下:
string str1=””;
string str2=””;
str1与str2的引⽤会是相同的也就是str1与str2在栈上保存的地址上相同的。请看下图
上图是str1的地址。
上图是str2的地址
4、如果⾮要说””与string.Empty有什么不同的话,我觉得1是写法不⼀样,string.Empty看起来好看~!~。2是在优化⽅⾯稍有差别。string.Empty于c#对””在语法级别的优化。这点可以通过string.Empty的内部实现看出来。
public static readonly String Empty = "";
也就是说””是通过CLR进⾏优化的,CLR会维护⼀个字符串池,以防在堆中创建重复的字符串。⽽string.Empty是⼀种c#语法级别的优化,是在C#编译器将代码编译为IL(即MSIL)时进⾏了优化,即所有对string类的静态字段Empty的访问都会被指向同⼀引⽤,以节省内存空间。
所以对””的优化更依赖CLR。
给⼤家看⼀下⼆者编译后的IL代码:
string str1=””; 编译后如下:
ldstr "" //从字符串池中取出⼀个””(实际上取的是地址)
stfld string ClassLibrary1.Class1::str1 //将””赋给str1(实际上赋的是地址)
string str2=string.Empty; 编译后如下:空字符串是什么
ldsfld string [mscorlib]System.String::Empty//取得string类的静态字段Empty(实际上取的是地址)
stfld string ClassLibrary1.Class1::str2//将Empty赋给str2(实际上赋的是地址)
总结:说了这么⼀⼤推我⾃⼰都觉得罗嗦,⽽且初学者朋友可能会看不懂。本⼈语⽂学的不好,表达能⼒⼀般还请⼤家谅解,下⾯我会挑要害来说。
“”与string.Empty在⽤法与性能上基本没区别。string.Empty是在语法级别对””的优化。
⼆、string.Empty与null的区别
因为string.Empty与””基本是⼀样的,所以string.Empty与null的区别也就代表了””与null的区别。
1、那就是string.Empty会在堆上占⽤⼀个长度为0的空间,⽽null不会。具体内容如下:
string str1=””;
string str2=null;
如刚才所说str1会在栈上保存⼀个地址,这个地址占4字节,指向内存堆中的某个长度为0的空间,这个空间保存的是str1的实际值。
str2同样会在栈上保存⼀个地址,这个地址也占4字节,但是这个地址是没有明确指向的,它哪也不指,其内容为
0x00000000。如下图
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论