最新文章专题视频专题问答1问答10问答100问答1000问答2000关键字专题1关键字专题50关键字专题500关键字专题1500TAG最新视频文章推荐1 推荐3 推荐5 推荐7 推荐9 推荐11 推荐13 推荐15 推荐17 推荐19 推荐21 推荐23 推荐25 推荐27 推荐29 推荐31 推荐33 推荐35 推荐37视频文章20视频文章30视频文章40视频文章50视频文章60 视频文章70视频文章80视频文章90视频文章100视频文章120视频文章140 视频2关键字专题关键字专题tag2tag3文章专题文章专题2文章索引1文章索引2文章索引3文章索引4文章索引5123456789101112131415文章专题3
问答文章1 问答文章501 问答文章1001 问答文章1501 问答文章2001 问答文章2501 问答文章3001 问答文章3501 问答文章4001 问答文章4501 问答文章5001 问答文章5501 问答文章6001 问答文章6501 问答文章7001 问答文章7501 问答文章8001 问答文章8501 问答文章9001 问答文章9501
当前位置: 首页 - 科技 - 知识百科 - 正文

.NET中可空值类型【Nullable<T>】实现原理

来源:懂视网 责编:小采 时间:2020-11-27 22:35:53
文档

.NET中可空值类型【Nullable<T>】实现原理

.NET中可空值类型【Nullable<T>】实现原理:为了让.Net中的值类型可以赋值为null,微软特地添加了Nullable<T>类型,也可简写为T?。但是Nullable<T>自身是结构体,也是值类型,那么它是如何实现将null赋值给值类型的呢? 下面通过自定义一个可空值类型来讲解Nullable<T
推荐度:
导读.NET中可空值类型【Nullable<T>】实现原理:为了让.Net中的值类型可以赋值为null,微软特地添加了Nullable<T>类型,也可简写为T?。但是Nullable<T>自身是结构体,也是值类型,那么它是如何实现将null赋值给值类型的呢? 下面通过自定义一个可空值类型来讲解Nullable<T

为了让.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());
 }
}

可以看到,变量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);
}

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

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

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

//因为Object类中的GetType方法不允许子类重写(避免子类隐藏自己的实际类型)
//所以这里使用关键字new来隐藏Object类中的GetType方法
public new Type GetType()
{
 return innerValue.GetType();
}

结论:没有可为空的值类型

至此,我们已经自定义了一个可为空的值类型XfhNullable<T>,通过以上代码,我们不难发现所谓可为空的值类型是不存在的,它是通过属性HasValue来对null值进行标记的,其内部通过字段innerValue(该字段对应Nullable<T>中的value字段)来维护该类型的值,若被赋值为null则innerValue初始化为值类型的初始值。换句话说,Nullable<T>只是在逻辑层面上实现了把null赋值给值类型,给我们一种值类型可为null的感觉。

最后说下可空值类型的装箱与拆箱。

CLR在对Nullable<T>实例执行装箱操作时首先检查它是否为null,若是则CLR不装箱任何东西而是直接返回null;若实例的值不是null则获取该实例的值(Value属性)并对这个值进行装箱操作。

拆箱时,对于null则返回一个Nullable<T>()实例,对于一个具体的数值,如5,则返回Nullable<T>(5)实例。

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。TEL:177 7030 7066 E-MAIL:11247931@qq.com

文档

.NET中可空值类型【Nullable<T>】实现原理

.NET中可空值类型【Nullable<T>】实现原理:为了让.Net中的值类型可以赋值为null,微软特地添加了Nullable<T>类型,也可简写为T?。但是Nullable<T>自身是结构体,也是值类型,那么它是如何实现将null赋值给值类型的呢? 下面通过自定义一个可空值类型来讲解Nullable<T
推荐度:
标签: 原理 类型 tt
  • 热门焦点

最新推荐

猜你喜欢

热门推荐

专题
Top