【備戰面試之】四、可空類型Nullable<T>到底是什麼鬼

来源:http://www.cnblogs.com/zhaopei/archive/2016/05/30/What_Is_Nullable.html
-Advertisement-
Play Games

首先我們都知道引用類型預設值都是null,而值類型的預設值都有非null。為什麼引用類型可以為空?因為引用類型變數都是保存一個對象的地址引用(就像一個url對應一個頁面),而引用類型值為null的時候是變數值指向了一個空引用(如同一個空的url)那為什麼值不能有空值呢?其實很簡單,因為如int值範圍... ...


值類型為什麼不可以為空

首先我們都知道引用類型預設值都是null,而值類型的預設值都有非null。

為什麼引用類型可以為空?因為引用類型變數都是保存一個對象的地址引用(就像一個url對應一個頁面),而引用類型值為null的時候是變數值指向了一個空引用(如同一個空的url

那為什麼值不能有空值呢?其實很簡單,因為如int值範圍是-2147483648到2147483647。其中根本就沒有給null值留那麼一個位置。

我們為什麼需要用到可空類型

舉個慄子吧,我們定義一個人(Person),它有三個屬性出生日期(BeginTime)、死亡日期(EndTime)、年齡(Age)。

如果這個人還健在人世,請問怎麼給死亡日期賦值?有人很聰明說“為空啊”。是的,這就是我們的需求。

微軟在C#2.0的時候就為我們引入了可null值類型( System.Nullable<T> ),那麼下麵來定義Person類。

 1 public class Person
 2 {
 3     /// <summary>
 4     /// 出生日期
 5     /// </summary>
 6     public DateTime BeginTime { get; set; }
 7     /// <summary>
 8     /// 死亡日期
 9     /// </summary>
10     public System.Nullable<DateTime> EndTiem { get; set; }
11     public int Age
12     {
13         get
14         {
15             if (EndTiem.HasValue)//如果掛了(如果有值,證明死了)
16             {
17                 return (EndTiem.Value - BeginTime).Days;
18             }
19             else//還沒掛
20             {
21                 return (DateTime.Now - BeginTime).Days;
22             }
23         }
24     }
25 }

 

這樣,我們就可以很容易獲得一個人的年齡了。

static void Main(string[] args)
{
    Person p1 = new Person()
    {
        BeginTime = DateTime.Parse("1990-07-19")
    };

    Person p2 = new Person()
    {
        BeginTime = DateTime.Parse("1893-12-26"),
        EndTiem = DateTime.Parse("1976-09-09")
    };

    Console.WriteLine("我今年" + p1.Age + "歲。");
    Console.WriteLine("毛爺爺活了" + p2.Age + "歲。");

    Console.ReadKey();
}

可空類型的實現

我們前面用到了 System.Nullable<DateTime> 來表示可空時間類型,其實平時我們用得更多的是 DateTime? 直接在類型T後面加一個問號,這兩種是等效的。多虧了微軟的語法糖。

我們來看看 System.Nullable<T> 到底是何物。

搜噶,原來是一個結構。還看到了我們屬性的 HasValue和Value屬性。原來竟這般簡單。一個結構兩個屬性,一個存值,一個存是否有值。那麼下麵我們也來試試吧。

不好意思,讓大家失望了。前面我們就說過了,值類型是不可以賦值null的(結構也是值類型)。

怎麼辦!怎麼辦!不對啊,微軟自己也是定義的結構,它怎麼可以直接賦值null呢。(奇怪,奇怪,畢竟是人家微軟自己搞得,可能得到了特殊的待遇吧)

可是,這樣就讓我們止步了嗎?NO!我們都知道,看微軟的IL(中間語言)的時候,就像脫了它的衣服一樣,很多時候不明白的地方都可以看個究竟,下麵我們就去脫衣服。

首先,我們用幾種不同的方式給可空類型賦值。

static void Main(string[] args)
{

    System.Nullable<int> number1 = null;

    System.Nullable<int> number2 = new System.Nullable<int>();

    System.Nullable<int> number3 = 23;

    System.Nullable<int> number4 = new System.Nullable<int>(88);

    Console.ReadKey();
}    

 

然後用reflector看編譯後的IL。

原來如此,可空類型的賦值直接等效於構造實例。賦null時其實就是調用空構造函數,有值時就就把值傳入帶參數的構造函數。(柳暗花明又一村。如此,我們是否可以接著上面截圖中的 MyNullable<T> 繼續模擬可空類型呢?且繼續往下看。)

public struct MyNullable<T> where T : struct
{
    //錯誤    1    結構不能包含顯式的無參數構造函數 
    //還好 bool預設值就是false,所以這裡不顯示為 this._hasValue = false也不會有影響
    //public MyNullable()
    //{
    //    this._hasValue = false;
    //}
    public MyNullable(T value)//有參構造函數
    {
        this._hasValue = true;
        this._value = value;
    }

    private bool _hasValue;

    public bool HasValue//是否不為空
    {
        get { return _hasValue; }
    }

    private T _value;
    public T Value//
    {
        get
        {
            if (!this._hasValue)//如沒有值,還訪問就拋出異常
            {
                throw new Exception(" 可為空的對象必須具有一個值");
            }
            return _value;
        }
    }
}

 

喲西,基本上已經模擬出了可空類型出來的。(但是我們還是不能直接賦值,只能通過構造函數的方式來使用自定義的可空類型)。

全部代碼如下:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace 可空類型
{
    public class Person
    {
        /// <summary>
        /// 出生日期
        /// </summary>
        public DateTime BeginTime { get; set; }
        /// <summary>
        /// 死亡日期
        /// </summary>
        public MyNullable<DateTime> EndTiem { get; set; } //這裡改用MyNullable
        /// <summary>
        /// 年齡
        /// </summary>
        public double Age
        {
            get
            {
                if (EndTiem.HasValue)//如果掛了(如果有值,證明死了)
                {
                    return (EndTiem.Value - BeginTime).Days / 365;
                }
                else//還沒掛
                {
                    return (DateTime.Now - BeginTime).Days / 365;
                }
            }
        }
    }

    public struct MyNullable<T> where T : struct
    {
        //錯誤    1    結構不能包含顯式的無參數構造函數 
        //還好 bool預設值就是false,所以這裡不顯示為 this._hasValue = false也不會有影響
        //public MyNullable()
        //{
        //    this._hasValue = false;
        //}
        public MyNullable(T value)//有參構造函數
        {
            this._hasValue = true;
            this._value = value;
        }

        private bool _hasValue;

        public bool HasValue//是否不為空
        {
            get { return _hasValue; }
        }

        private T _value;
        public T Value//
        {
            get
            {
                if (!this._hasValue)//如沒有值,還訪問就拋出異常
                {
                    throw new Exception(" 可為空的對象必須具有一個值");
                }
                return _value;
            }
        }
    }
    class Program
    {
        static void Main(string[] args)
        {
            Person p1 = new Person()
            {
                BeginTime = DateTime.Parse("1990-07-19")
            };

            Person p2 = new Person()
            {
                BeginTime = DateTime.Parse("1893-12-26"),
                EndTiem = new MyNullable<DateTime>(DateTime.Parse("1976-09-09"))//這裡使用MyNullable的有參構造函數
            };

            Console.WriteLine("我今年" + p1.Age + "歲。");
            Console.WriteLine("毛爺爺活了" + p2.Age + "歲。");

            Console.ReadKey();
        }

    }
}
View Code

 

和系統的可空類型得出了相同的結果。

總結

  • 可空類型是結構(也就是值類型)
  • 所以可空類型的null值和引用類型的null是不一樣的。(可空類型的並不是引用類型的null,而是用結構的另一種表示方式來表示null

 

有同學問,怎麼樣才可以做到直接賦值呢?這個我也沒有什麼好的辦法,或許需要編譯器的支持。

以上內容都是胡說八道。希望能對您有那麼一點點用處,感謝閱讀。

(首發鏈接:http://www.cnblogs.com/zhaopei/p/5537759.html )

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 現在很多用戶被資料庫的慢的問題所困擾,又苦於花錢請一個專業的DBA成本太高。軟體維護人員對資料庫的瞭解又不是那麼深入,所以導致問題遲遲不能解決,或只能暫時解決不能得到根治。開發人員解決數據問題基本又是搜遍百度各種方法嘗試個遍,可能錯過診斷問題的最佳時機又可能嘗試一堆方法最後無奈放棄。 怎麼樣讓瑣事纏 ...
  • 一般,我們看到術語“索引”和“鍵”交換使用,但實際上這兩個是不同的。索引是存儲在資料庫中的一個物理結構,鍵純粹是一個邏輯概念。鍵代表創建來實施業務規則的完整性約束。索引和鍵的混淆通常是由於資料庫使用索引來實施完整性約束。 接下來我們看看資料庫中的主鍵約束、唯一鍵約束和唯一索引的區別。 SQL> se ...
  • MongoDB是一個開源的文檔資料庫,支持高性能、高可用性、自動縮放。 在MongoDB中,一條記錄就是一個文檔,是由欄位和值對構成一個數據結構,類似於JSON對象。欄位的值可以包括其他文檔、數組和文檔的數組。 數據結構如下所示: ...
  • 腳本最好都放在/usr/local/sbin中 腳本的執行 sh -x 腳本.sh -x可以查看執行過程 1在腳本中使用變數 使用變數的時候,需要使用$符號: #!/bin/bash ##把命令賦值為變數,需要使用反引號 d=`date +"%H:%M:%S"` echo "The script b ...
  • grep是UNIX和LINUX中使用最廣泛的命令之一。grep允許對文本文件進行模式查找。如果找到匹配模式, grep列印包含模式的所有行。grep支持基本正則表達式,也支持其擴展集。grep有三種變形,即: grep:標準grep命令,這裡主要討論此格式; Egrep:等同於grep -E,擴展g ...
  • 算術運算符 + - * / % 表示加減乘除和取餘運算+= -= *= /= 同 C 語言中的含義 位操作符 > >>= 表示位左右移一位操作& &= | |= 表示按位與、位或操作~ ! 表示非操作^ ^= 表示異或操作 關係運算符 = == != 表示大於、小於、大於等於、小於等於、等於、不等於 ...
  • OSI 七層模型通過七個層次化的結構模型使不同的系統不同的網路之間實現可靠的通訊,因此其最主要的功能就是幫助不同類型的主機實現數據傳輸 。 OSI 七層模型通過七個層次化的結構模型使不同的系統不同的網路之間實現可靠的通訊,因此其最主要的功能就是幫助不同類型的主機實現數據傳輸 。 完成中繼功能的節點通 ...
  • 上一篇我們已經可以獲取各種FileHandler的實例和對應的元數據。本篇,我們做一個稍微完整的文件管理器。 1、修改介面IFileHandler,傳入文件名 2、修改具體的FileHandler。 3、修改主函數 運行結果: 可以看到,對每一個具體的文件,均找到了正確的處理實例進行處理。avi文件 ...
一周排行
    -Advertisement-
    Play Games
  • 概述:本文代碼示例演示瞭如何在WPF中使用LiveCharts庫創建動態條形圖。通過創建數據模型、ViewModel和在XAML中使用`CartesianChart`控制項,你可以輕鬆實現圖表的數據綁定和動態更新。我將通過清晰的步驟指南包括詳細的中文註釋,幫助你快速理解並應用這一功能。 先上效果: 在 ...
  • openGauss(GaussDB ) openGauss是一款全面友好開放,攜手伙伴共同打造的企業級開源關係型資料庫。openGauss採用木蘭寬鬆許可證v2發行,提供面向多核架構的極致性能、全鏈路的業務、數據安全、基於AI的調優和高效運維的能力。openGauss深度融合華為在資料庫領域多年的研 ...
  • openGauss(GaussDB ) openGauss是一款全面友好開放,攜手伙伴共同打造的企業級開源關係型資料庫。openGauss採用木蘭寬鬆許可證v2發行,提供面向多核架構的極致性能、全鏈路的業務、數據安全、基於AI的調優和高效運維的能力。openGauss深度融合華為在資料庫領域多年的研 ...
  • 概述:本示例演示了在WPF應用程式中實現多語言支持的詳細步驟。通過資源字典和數據綁定,以及使用語言管理器類,應用程式能夠在運行時動態切換語言。這種方法使得多語言支持更加靈活,便於維護,同時提供清晰的代碼結構。 在WPF中實現多語言的一種常見方法是使用資源字典和數據綁定。以下是一個詳細的步驟和示例源代 ...
  • 描述(做一個簡單的記錄): 事件(event)的本質是一個委托;(聲明一個事件: public event TestDelegate eventTest;) 委托(delegate)可以理解為一個符合某種簽名的方法類型;比如:TestDelegate委托的返回數據類型為string,參數為 int和 ...
  • 1、AOT適合場景 Aot適合工具類型的項目使用,優點禁止反編 ,第一次啟動快,業務型項目或者反射多的項目不適合用AOT AOT更新記錄: 實實在在經過實踐的AOT ORM 5.1.4.117 +支持AOT 5.1.4.123 +支持CodeFirst和非同步方法 5.1.4.129-preview1 ...
  • 總說周知,UWP 是運行在沙盒裡面的,所有許可權都有嚴格限制,和沙盒外交互也需要特殊的通道,所以從根本杜絕了 UWP 毒瘤的存在。但是實際上 UWP 只是一個應用模型,本身是沒有什麼許可權管理的,許可權管理全靠 App Container 沙盒控制,如果我們脫離了這個沙盒,UWP 就會放飛自我了。那麼有沒... ...
  • 目錄條款17:讓介面容易被正確使用,不易被誤用(Make interfaces easy to use correctly and hard to use incorrectly)限制類型和值規定能做和不能做的事提供行為一致的介面條款19:設計class猶如設計type(Treat class de ...
  • title: 從零開始:Django項目的創建與配置指南 date: 2024/5/2 18:29:33 updated: 2024/5/2 18:29:33 categories: 後端開發 tags: Django WebDev Python ORM Security Deployment Op ...
  • 1、BOM對象 BOM:Broswer object model,即瀏覽器提供我們開發者在javascript用於操作瀏覽器的對象。 1.1、window對象 視窗方法 // BOM Browser object model 瀏覽器對象模型 // js中最大的一個對象.整個瀏覽器視窗出現的所有東西都 ...