string、Empty和null三者的区别

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了string、Empty和null三者的区别相关的知识,希望对你有一定的参考价值。

参考技术A

  这是一个及其常见的问题 网上已经有关于这个问题的很多讨论 但是我觉得都是不求甚解 有一些还是在误导别人 下面我来说下我对这三者的理解 如有错误的地方请大家及时指正

  一 与string Empty我认为是一样的 网上有一篇被转载了几十遍的文章是这样说的string Empty 不分配存储空间 分配一个长度为空的存储空间 我认为这句话是错误并且含糊不清的

   实际上Empty是string类中的一个静态的只读字段 他的定义是这样的

  public static readonly String Empty = ;

  也就是说string Empty的内部实现是等于 的

   我要反驳string Empty 不分配存储空间 分配一个长度为空的存储空间这个观点 首先string Empty与 都会分配存储空间 具体的说是都会在内存的栈和堆上分配存储空间

  有一点先说明一下 引用类型是将对象是实际数据保存在堆上 将对象在堆上的地址保存在栈上 因此string Empty与 都会在栈上保存一个地址这个地址占 字节 指向内存堆中的某个长度为 的空间 这个空间保存的是string Empty的实际值 这个我可以用VS 跟踪下内存给大家演示

  

  上图中的 x e 即是变量str在栈中存储的地址

  对于 请看下图

  

  这个图的效果跟上图是一样的 也就是说 也是在栈上保存了一个地址

   CLR会对字符串进行优化 所以 和string Empty也都会被优化

  声明如下两个变量

  string str = ;

  string str = ;

  str 与str 的引用会是相同的也就是str 与str 在栈上保存的地址上相同的 请看下图

     上图是str 的地址                                      上图是str 的地址

  可见str 于str 的地址是相同的 也就是说str 会在堆上占用一个长度为 的空间 而str 不会在堆上开辟新的空间 str 于str 在堆上共享同一空间

  同理对于

  string str =string Empty;

  string str = string Empty;

  str 与str 的地址也是相同的 也是在内存堆上共享同一空间 如下图

  上图是str 的地址                                上图是str 的地址

   如果非要说 与string Empty有什么不同的话 我觉得 是写法不一样 string Empty看起来好看~!~ 是在优化方面稍有差别 string Empty于c#对 在语法级别的优化 这点可以通过string Empty的内部实现看出来

  public static readonly String Empty = ;

  也就是说 是通过CLR进行优化的 CLR会维护一个字符串池 以防在堆中创建重复的字符串 而string Empty是一种c#语法级别的优化 是在C#编译器将代码编译为IL(即MSIL)时进行了优化 即所有对string类的静态字段Empty的访问都会被指向同一引用 以节省内存空间

  所以对 的优化更依赖CLR

  给大家看一下二者编译后的IL代码:

  string  str = ;  编译后如下

  ldstr               //从字符串池中取出一个 (实际上取的是地址)

  stfld      string ClassLibrary Class ::str    //将 赋给str (实际上赋的是地址)

  string  str =string Empty; 编译后如下

  ldsfld     string [mscorlib]System String::Empty//取得string类的静态字段Empty(实际上取的是地址)

  stfld      string ClassLibrary Class ::str //将Empty赋给str (实际上赋的是地址)

  总结 说了这么一大推我自己都觉得罗嗦 而且初学者朋友可能会看不懂 本人语文学的不好 表达能力一般还请大家谅解 下面我会挑要害来说

   与string Empty在用法与性能上基本没区别 string Empty是在语法级别对 的优化

  二 string Empty与null的区别

  因为string Empty与 基本是一样的 所以string Empty与null的区别也就代表了 与null的区别

     那就是string Empty会在堆上占用一个长度为 的空间 而null不会 具体内容如下

  string str = ;

  string str =null;

  如刚才所说str 会在栈上保存一个地址 这个地址占 字节 指向内存堆中的某个长度为 的空间 这个空间保存的是str 的实际值

  str 同样会在栈上保存一个地址 这个地址也占 字节 但是这个地址是没有明确指向的 它哪也不指 其内容为 x 如下图

lishixinzhi/Article/program/net/201311/13583

如何在流利的断言中比较 null 和 string.Empty(或“”)?

【中文标题】如何在流利的断言中比较 null 和 string.Empty(或“”)?【英文标题】:How can I compare null and string.Empty (or "") in fluent assertions? 【发布时间】:2014-07-24 21:28:43 【问题描述】:

我有两个相同类型的对象,该类型有一个字符串字段,在第一个对象中值为空,在第二个对象中值为“”,我如何强制流畅的断言假设这是正确的?

断言本身:

callResult.ShouldBeEquivalentTo(centreResponse, 
                                opt => opt.Excluding(r => r.DateOpen));

这里抛出异常,说明期望值为null,真实值为""(反之亦然)

【问题讨论】:

对象是什么类型?我可以假设中心响应是一个字符串,而 opt 是 string[] 还是什么? 为什么要将null 和空字符串视为等效?这不是一个好主意。 对象是 CentreResponse 类型,它是一个复杂类型,关于比较规则 - 这是要求,它们来自不同的数据源,但在这个特定的上下文中 null 和空字符串是相等的 加上field,你的意思是C#字段吗? 我的意思是一个字符串类型的属性,抱歉误导:) 【参考方案1】:

您可以做的是像这样覆盖string 类型的属性的行为:

callResult.ShouldBeEquivalentTo(centreResponse, opt => opt
          .Excluding(r => r.DateOpen)
          .Using<string>(ctx => CompareStrings(ctx)).WhenTypeIs<string>());       

public void CompareStrings(IAssertionContext<string> ctx)

    var equal = (ctx.Subject ?? string.Empty).Equals(ctx.Expectation ?? string.Empty);

    Execute.Assertion
        .BecauseOf(ctx.Because, ctx.BecauseArgs)
        .ForCondition(equal)
        .FailWith("Expected context:string to be 0reason, but found 1", ctx.Subject, ctx.Expectation);
    

您可以通过将CompareStrings 方法封装在IAssertionRule 的实现中来稍微清理一下。请参阅 FluentAssertions here 的单元测试中的 RelaxingDateTimeAssertionRule

您可以将该自定义断言规则添加为您的 callResult 变量类型的所有断言的默认值,但我仍然需要添加一些内容以允许全局默认值。

【讨论】:

以上是关于string、Empty和null三者的区别的主要内容,如果未能解决你的问题,请参考以下文章

stringEmpty和null三者的区别

@NotEmpty@NotNull 和 @NotBlank 的区别

数据库建字段,默认值空和empty string有啥区别

C#中null""string.empty区别

C# string总结

C# string总结