下面是一些代码,它们演示了我无法将结构类型声明和初始化为null。 Nullable类型是一个结构,为什么我能够将它设置为null?
Nullable<bool> b = null;
if (b.HasValue)
{
Console.WriteLine("HasValue == true");
}
//Does not compile...
Foo f = null;
if (f.HasValue)
{
Console.WriteLine("HasValue == true");
}
Foo
定义为
public struct Foo
{
private bool _hasValue;
private string _value;
public Foo(string value)
{
_hasValue = true;
_value = value;
}
public bool HasValue
{
get { return _hasValue; }
}
public string Value
{
get { return _value; }
}
}
问题已得到解答(见下文)。为了澄清我会发布一个例子。 C#代码:
using System;
class Program
{
static void Main(string[] args)
{
Nullable<bool> a;
Nullable<bool> b = null;
}
}
产生以下IL:
.method private hidebysig static void Main(string[] args) cil managed
{
.entrypoint
// Code size 10 (0xa)
.maxstack 1
.locals init ([0] valuetype [mscorlib]System.Nullable`1<bool> a,
[1] valuetype [mscorlib]System.Nullable`1<bool> b)
IL_0000: nop
IL_0001: ldloca.s b
IL_0003: initobj valuetype [mscorlib]System.Nullable`1<bool>
IL_0009: ret
} // end of method Program::Main
声明了a和b,但只初始化了b。
答案 0 :(得分:23)
C#编译器为你提供了一些糖,所以你真的这样做:
Nullable<bool> b = new Nullable<bool>();
这是句法糖
bool? b = null;
if (b ?? false)
{
b = true;
}
答案 1 :(得分:13)
C#有一些语法糖,可以让你看似将可为空的类型设置为null
。您实际做的事情是将可空类型的HasValue
属性设置为false
。
答案 2 :(得分:3)
因为您实际上并未将Nullable<T>
变量设置为null
。结构仍然存在。它通过结构中的内部位标志表示null
。
还有一些编译器糖可以让魔术在幕后发生。
答案 3 :(得分:2)
您不能将结构设置为null,但您可以进行隐式类型转换,这就是幕后发生的事情。