.NET中可空值类型【NullableT】实现原理

为了让.Net中的值类型可以赋值为null,微软特地添加了Nullable<T>类型,也可简写为T?。但是Nullable<T>自身是结构体,也是值类型,那么它是如何实现将null赋值给值类型的呢?

下面通过自定义一个可空值类型来讲解Nullable<T>的实现原理。

自定义可空值类型

struct XfhNullable<T> where T : struct { private T innerValue; //这个属性很重要 public bool HasValue { set; get; } public T Value { get { return HasValue ? innerValue: throw new InvalidOperationException(); } } public XfhNullable(T value) { this.innerValue= value; HasValue = true; } public T GetValueOrDefault(T value) { return HasValue ? this.innerValue: value; } public T GetValueOrDefault() { return this.innerValue; } }

一个可空值类型的结构体大致功能已经定义好了,下面我们来创建可空值类型的实例来验证下。

using static System.Console; class Program { static void Main() { //使用结构体默认的无参构造函数进行实例化 XfhNullable<int> num = new XfhNullable<int>(); WriteLine(num.HasValue); WriteLine(null_num.GetValueOrDefault()); } }

.NET中可空值类型【NullableT】实现原理

可以看到,变量num并不含有值,调用GetValueOrDefault()则会获取它的默认值 0;

这时我们将null赋值给变量num会发现编译器报错Cannot convert null to 'XfhNullable<int>' because it is a non-nullable value type这是因为编译器把我们定义的结构体XfhNullable<T>看作是普通值类型而非可空值类型,所以我们还要添加可空值类型和XfhNullable<T>之间的转换功能。

public static implicit operator XfhNullable<T>(T? nullabelValue) { if (nullabelValue== null) { return new XfhNullable<T>(); } return new XfhNullable<T>(nullabelValue.Value); }

上面的代码实现了可空值类型向XfhNullable<T>的隐式转换,添加上面代码之后发现编译器不再报错。XfhNullable<T>已经成为一个可为null的值类型。

static void Main() { XfhNullable<int> null_num = null; WriteLine(null_num.HasValue); }

XfhNullable<T>中的属性HasValue的作用就是标记当前类型是否为null,若是则返回False,否则返回True。当HasValue为False时调用该类型的Value属性则会抛出异常InvalidOperationException。但可调用GetValueOrDefault()方法来获取类型的默认值。

Nullable<T>类型可以通过运算符==来判断值是否为null,我们也可以通过运算符重载来实现该功能:

public static bool operator ==(XfhNullable<T> cn, object obj) { if (cn.HasValue) { return false; } return true; } public static bool operator !=(XfhNullable<T> cn, object obj) { return !(cn == obj); }

static void Main() { XfhNullable<int> null_num = null; WriteLine(null_num == null); }

接下来,我们来实现普通值类型和XfhNullable<T>之间的转换:

public static implicit operator XfhNullable<T>(T value) { return new XfhNullable<T>(value); } public static explicit operator T(XfhNullable<T> value) { return value.innerValue; }

static void Main() { XfhNullable<int> null_num = null; null_num = 12;//int类型隐式转换为XfhNullable<int>类型 WriteLine(null_num == null); WriteLine(null_num.Value); int i = (int)null_num;//XfhNullable<int>类型强制转换为int类型 WriteLine(i); }

.NET中可空值类型【NullableT】实现原理

获取实例在运行时的类型:

static void Main() { XfhNullable<int> null_num = 12; WriteLine(null_num.GetType()); }

这个返回值不大友好,我们希望这里返回内置的值类型,System.Int32,具体实现代码如下:

内容版权声明:除非注明,否则皆为本站原创文章。

转载注明出处:https://www.heiqu.com/wdpxzd.html