Nullable Value Type

      在.net2.0以前,值类型是不充许赋值为null的,这也是值类型和引用类型的一个重要区别。随着.net2.0的出现,这一局面得以改善,这种功能有点像数据库中的数据类型(int,datetime等),在数据库中列都可以设置是否可为Null。C#实现这种功能主要是依靠一个重要的结构:Nullable<T>,我们先看下它的实现:

[Serializable]
    
public   struct  Nullable < T >   where  T :  struct
    {
        
//
        
//  摘要:
        
//      Initializes a new instance of the System.Nullable<T> structure to the specified
        
//      value.
        
//
        
//  参数:
        
//    value:
        
//      A value type.
         public  Nullable(T value);

        
public   static   explicit   operator  T(T ?  value);
        
public   static   implicit   operator  T ? (T value);

        
//  摘要:
        
//      Gets a value indicating whether the current System.Nullable<T> object has
        
//      a value.
        
//
        
//  返回结果:
        
//      true if the current System.Nullable<T> object has a value; false if the current
        
//      System.Nullable<T> object has no value.
         public   bool  HasValue {  get ; }
        
//
        
//  摘要:
        
//      Gets the value of the current System.Nullable<T> value.
        
//
        
//  返回结果:
        
//      The value of the current System.Nullable<T> object if the System.Nullable<T>.HasValue
        
//      property is true. An exception is thrown if the System.Nullable<T>.HasValue
        
//      property is false.
        
//
        
//  异常:
        
//    System.InvalidOperationException:
        
//      The System.Nullable<T>.HasValue property is false.
         public  T Value {  get ; }

        
//  摘要:
        
//      Indicates whether the current System.Nullable<T> object is equal to a specified
        
//      object.
        
//
        
//  参数:
        
//    other:
        
//      An object.
        
//
        
//  返回结果:
        
//      true if the other parameter is equal to the current System.Nullable<T> object;
        
//      otherwise, false. This table describes how equality is defined for the compared
        
//      values: Return Value Description true The System.Nullable<T>.HasValue property
        
//      is false, and the other parameter is null. That is, two null values are equal
        
//      by definition.  -or- The System.Nullable<T>.HasValue property is true, and
        
//      the value returned by the System.Nullable<T>.Value property is equal to the
        
//      other parameter.  false The System.Nullable<T>.HasValue property for the
        
//      current System.Nullable<T> structure is true, and the other parameter is
        
//      null.  -or- The System.Nullable<T>.HasValue property for the current System.Nullable<T>
        
//      structure is false, and the other parameter is not null.  -or- The System.Nullable<T>.HasValue
        
//      property for the current System.Nullable<T> structure is true, and the value
        
//      returned by the System.Nullable<T>.Value property is not equal to the other
        
//      parameter.
         public   override   bool  Equals( object  other);
        
//
        
//  摘要:
        
//      Retrieves the hash code of the object returned by the System.Nullable<T>.Value
        
//      property.
        
//
        
//  返回结果:
        
//      The hash code of the object returned by the System.Nullable<T>.Value property
        
//      if the System.Nullable<T>.HasValue property is true, or zero if the System.Nullable<T>.HasValue
        
//      property is false.
         public   override   int  GetHashCode();
        
//
        
//  摘要:
        
//      Retrieves the value of the current System.Nullable<T> object, or the object's
        
//      default value.
        
//
        
//  返回结果:
        
//      The value of the System.Nullable<T>.Value property if the System.Nullable<T>.HasValue
        
//      property is true; otherwise, the default value of the current System.Nullable<T>
        
//      object. The type of the default value is the type argument of the current
        
//      System.Nullable<T> object, and the value of the default value consists solely
        
//      of binary zeroes.
         public  T GetValueOrDefault();
        
//
        
//  摘要:
        
//      Retrieves the value of the current System.Nullable<T> object, or the specified
        
//      default value.
        
//
        
//  参数:
        
//    defaultValue:
        
//      A value to return if the System.Nullable<T>.HasValue property is false.
        
//
        
//  返回结果:
        
//      The value of the System.Nullable<T>.Value property if the System.Nullable<T>.HasValue
        
//      property is true; otherwise, the defaultValue parameter.
         public  T GetValueOrDefault(T defaultValue);
        
//
        
//  摘要:
        
//      Returns the text representation of the value of the current System.Nullable<T>
        
//      object.
        
//
        
//  返回结果:
        
//      The text representation of the value of the current System.Nullable<T> object
        
//      if the System.Nullable<T>.HasValue property is true, or an empty string ("")
        
//      if the System.Nullable<T>.HasValue property is false.
         public   override   string  ToString();

  

       Nullable<T>数据类型:Nullable<T>能够使C#中的值类型赋值为null,但同时它本身是一个值类型。这个结构的实例比较小,在存储上依然可以存储在堆栈上。        

            Nullable < int >  i  =   1 ;
            Nullable
< int >  j  =   null ;

 

      生成的IL代码足以说明Nullable<T>本身是值类型:

.locals init ([ 0 ] valuetype [mscorlib]System.Nullable` 1 < int32 >  i,
[
1 ] valuetype [mscorlib]System.Nullable` 1 < int32 >  j)

  

      Nullable<T>的取值,既然能够充许值类型为null,那么如何读取值呢?

      第一:Nullable<T>属性
            1:HasValue:获取一个值,指示当前的 Nullable<T> 对象是否有值。
            2:Value:获取当前的 Nullable<T> 值。如果hasValue等于false,则会抛出异常。

      第二:Nullable<T>中包含两个特别重要的方法:
            1:T GetValueOrDefault();检索当前 Nullable<T> 对象的值,或该对象的默认值。internal T value=default(T),这个私有字段可以保证返回一个相应类型的默认值。
            2:T GetValueOrDefault(T defaultValue);检索当前 Nullable<T> 对象的值或指定的默认值。

     Nullable<T>的别名:Nullable<int> 可以写成int?,这是为了方便书写等原因设计。下面的代码是等价的:
   

int ?  i = 3 ; // Nullable<int> i=3
int ? j = null ; // Nullable<int> j=null

  

      Nullable<T>对操作符的影响:

           1:一元操作符,如果操作数null,则结果返回null;
           2:二元操作符,如果其中一个操作符为null,结果返回null
           3:叛等:
             1>如果两个操作数都为null,返回true。
             2>如果有一个操作数为null,返回false。
             3>如果两个操作数都不为null,则比较两个操作数的Value。
          4:比较操作符:
            1>如果其中一个操作数为null,则返回false。
            2>如果两个都不为null,则比较两个操作数的Value。

      C#对Nullable<T>的应用:??操作符:如果 ?? 运算符的左操作数非 null,该运算符将返回左操作数,否则返回右操作数。它结合了判断对象是否为空的操作。下面的代码是等价的:可以看出代码简洁不少。
 

DateTime  k  =  j  ??  DateTime.Now ;
DateTime kk
= j == null   ? DateTime .Now :(DateTime )j ;

  

      Nullable Value Type的GetType方法,一个类型申明成Nullable<T>,我们理所当然的认为这个变量的类型会是Nullable<T>,但实际并非我们想象的那样。这里并不会返回Nullable(Int32),而是会返回System.Int32。CLR会直接返回T的类型,而不是Nullable<T>。

Int32 ?  i = 1 ;
i.GetType();

  

      Nullable Value Type对接口方法的影响:我们来看下Nullable<int> 和int在调用IComparable的区别。Nullable<int>并没有实现IComparable接口,但int有实现,所以我样想调用ToCompareTo方法可以这样写(CLR)对Nullable<T>有特殊处理:
   

Int32 ?  i  =   1 ;
Int32  iResult 
=  ((IComparable)i).CompareTo( 1 );

  

      如果没有CLR对Nullable<T>的特殊支持,我们需要这样写,明显可以看出是比较麻烦的方法:
  

 Int32 iResult2  =  ((IComparable)(Int32)i).CompareTo( 1 );

   
  
 

你可能感兴趣的:(value)