转:细说可空类型 nullable

可空类型是System.Nullable结构体的实列。一个可空类型代表了相应值类型的正确范围附加null值。这么说来,其实也不是很明子,命题嘛,一般不求易懂,但求准确。

那我就来说说这可空类型吧,上次说到了值类型与引用类型,其中就说到了,值类型是不能为空的,int i=null是错的,值类型为能为空,但我们有时候需要让值类型也可以为空,怎么办呢,那就在值类型的值的范围上扩充一个null值。

  1. 为什么要有可空类型
  2.  

我们在日常开发中,什么地方需要让值类型也变成可空呢?如果你有试过把数据库内的数据对象化的时候,面对数据库中的字段定义如:image

要转化成对象

public class XXXX
{
    public int id { get; set; }
    //……

}

这时候就出现问题了,在数据库中是允许为空的,但是在类设计的时候,int类型就是不允许为空的,你不给它赋值,它也是有默认值0的。怎么办?这与数据库的设计是有冲突的,于是,就必须使用咱们的可空类型,我个人理解这也就是为什么要有可空类型的原因。

怎么表示可空类型

使用可空类型,上面的类就可以写成

public class XXXX
{
    public int? id { get; set; }
    //……

}

没看错,就是在int后面加上?号,这就是可空类型的表示,当然还有其它的表示方式

Nullable<int> d = null;
 

这两种表示方式其实是等效的,就根据喜好和使用场景灵活使用就行。

怎么判断是null还是其它值

对于可空类型,我们可以很简单地使用null与可空类型进行比较,就可能判断变量的值是否为null。如:

Nullable<int> d = null;
bool isNull = d == null ? true : false; 

当然,我还有其它的办法与方法来做这同样效果的事,如:

Nullable<int> d = null;
bool isNull = !d.HasValue; 

注意,如果HasValue是为false时,使用 var result=d.Value时,会抛出异常System.InvalidOperationException

根据情况,根据需要,灵活使用以上两种判断方法。

可空类型怎么给值类型赋值

可空类型不可以直接转化成值类型,也不可以直接对值类型赋值,如:

Nullable<int> d = null;
int result = d;//错 
int result = (int)d;//错

那怎么做呢,最简单的办法就是

Nullable<int> d = null;
int result = d.GetValueOrDefault();

这时result的值为0;

或者也可以判断是否为空,对它进行赋值

Nullable<int> d = null;
int result;
if (!d.HasValue)
{
    result = d.Value;
}

如果可空类型是经过运算给值类型赋值。那就还有一种办法

Nullable<int> d = null;
int result = d ?? +1; 

即使用??进行转意。这是的result的值为1;

反射中的可空类型

我们经常会在反射中使用的可空类型,那怎么来使用可空类型呢?

public class NullableTest
{
    public int? ID { get; set; }
}
var propertyInfo= typeof(NullableTest).GetProperty("ID");

查询变量的值时,我们会发现

propertyInfo.PropertyType.Name Nullable`1
propertyInfo.PropertyType.IsGenericType true
propertyInfo.PropertyType.IsGenericTypeDefinition false
propertyInfo.PropertyType.GetGenericTypeDefinition().Name Nullable`1

我们根本无法使用一般泛型得到泛型基类型的方法GetGenericTypeDefinition()得到可空类型的基类型,此时,我们就需要使用GetGenericArguments()去得到泛型的基础类型。

var propertyInfo = typeof(NullableTest).GetProperty("ID");
if (propertyInfo.PropertyType.IsGenericType && propertyInfo.PropertyType.GetGenericTypeDefinition() == typeof(Nullable<>))
{
    Type[] typeArray = propertyInfo.PropertyType.GetGenericArguments();
    Type baseType = typeArray[0];
}

做到这里,我们就可以使用反射来完成任何我们想要的操作了。

原文地址:https://www.cnblogs.com/davinci/p/1652466.html