【備戰面試之】四、可空類型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
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...