深入理解c# checked unchecked 關(guān)鍵字
更新時間:2013年05月18日 16:13:07 作者:
本篇文章是對c#中的checked unchecked 關(guān)鍵字進行了詳細的分析介紹,需要的朋友參考下
checked 和 unchecked關(guān)鍵字用來限定檢查或者不檢查數(shù)學(xué)運算溢出的;如果使用了checked發(fā)生數(shù)學(xué)運算溢出時會拋出OverflowException;如果使用了unchecked則不會檢查溢出,算錯了也不會報錯。
1. 一段編譯沒通過的代碼
int a = int.MaxValue * 2;
以上代碼段編譯沒有通過,在VS2010中會有一條紅色的波浪線指出這段代碼有問題:”The operation overflows at compile time in checked mode”。這說明了編譯器會在編譯時檢查數(shù)學(xué)運算是否溢出。但是編譯時能檢查出溢出的情況僅限于使用常量的運算。2中的代碼編譯器就不報不出錯誤來了。
2. 一段編譯通過但是不能得到正確結(jié)果的代碼
int temp = int.MaxValue;
int a = temp * 2;
Console.Write(a);
我先把常量int.MaxValue的值給了臨時變量temp,然后使用臨時變量乘以2計算結(jié)果賦值給a;這段代碼是可以正常執(zhí)行的,執(zhí)行結(jié)果將輸出 -2。
這說明在運行時默認情況程序是不會檢查算術(shù)運算是否溢出的,cpu只管算,對于它來講按規(guī)則算就是了,結(jié)果對不對不是他的錯。
正常執(zhí)行了,而結(jié)果是錯誤的,這是非常危險的情況,該如何避免這種危險呢?請看3
3. 使用checked關(guān)鍵字,溢出時報警
int temp = int.MaxValue;
try
{
int a = checked(temp * 2);
Console.WriteLine(a);
}
catch (OverflowException)
{
Console.WriteLine("溢出了,要處理喲");
}
使用checked關(guān)鍵字修飾temp*2的計算結(jié)果,并使用try catch在發(fā)生溢出時做處理。以上代碼將輸出:“溢出了,要處理喲”
問題是如果一段代碼中有很多算術(shù)運算都需要做溢出檢查,那會有很多checked修飾的表達式,怎么辦呢?請看4
4. checked關(guān)鍵字可以修飾一個語句塊,請看下面代碼
int temp = int.MaxValue;
try
{
checked
{
int num = temp / 20;
int a = temp * 2;
int c = temp * 1000;
}
}
catch (OverflowException)
{
Console.WriteLine("溢出了,要處理喲");
}
以上程序輸出結(jié)果和3一樣
5. checked在避免算術(shù)溢出方面很有用,那么unchecked呢,它有用嗎?答案是肯定的,有時候我們不需要準確的計算結(jié)果,我們只是需要那么一個數(shù)而已,至于溢出不溢出的關(guān)系不大,比如說生成一個對象的HashCode,比如說根據(jù)一個算法計算出一個相對隨機數(shù),這都是不需要準確結(jié)果的。如下代碼片段
class Person
{
public string Name { get; set; }
public string Title { get; set; }
public override int GetHashCode()
{
return unchecked(Name.GetHashCode() + Title.GetHashCode());
}
}
unchecked也可以修飾語句塊,其用法和checked完全一樣。
6. checked和unchecked是可以嵌套使用的,雖然沒啥意義。語句是否是checked以最近嵌套的checked或者unchecked決定
7. 從IL中看checked關(guān)鍵字
C#代碼:
static void Main(string[] args)
{
int a = int.MaxValue;
int b = a * 2;
int c = checked(a * 2);
int d = unchecked(a + 3);
Console.Read();
}
對應(yīng)IL
.method private hidebysig static void Main(string[] args) cil managed
{
.entrypoint
// Code size 26 (0x1a)
.maxstack 2
.locals init ([0] int32 a,
[1] int32 b,
[2] int32 c,
[3] int32 d)
IL_0000: nop
IL_0001: ldc.i4 0x7fffffff
IL_0006: stloc.0
IL_0007: ldloc.0
IL_0008: ldc.i4.2
IL_0009: mul
IL_000a: stloc.1
IL_000b: ldloc.0
IL_000c: ldc.i4.2
IL_000d: mul.ovf
IL_000e: stloc.2
IL_000f: ldloc.0
IL_0010: ldc.i4.3
IL_0011: add
IL_0012: stloc.3
IL_0013: call int32 [mscorlib]System.Console::Read()
IL_0018: pop
IL_0019: ret
} // end of method Program::Main
請看IL中的紅色和綠色加重顯示代碼,可以看出使用checked時,IL的運算是mul.ovf;不使用checked或者使用unchecked時的IL運算函數(shù)是mul或者add,不帶.ovf。
8. checked或者unchecked只影響其包圍的語句,不會影響到包圍的語句內(nèi)調(diào)用函數(shù)的代碼塊,如下示例:
static void Main(string[] args)
{
int a = int.MaxValue;
int b = 20;
checked
{
int c = TestMethod(a, b);
Console.WriteLine(c);
}
}
static int TestMethod(int a, int b)
{
return a * b;
}
上面代碼將會正常執(zhí)行,checked語句塊并未起到應(yīng)有的作用。
9. 全局開啟或者關(guān)閉checked編譯選項
在項目屬性頁上選擇“生成”選項卡,然后點擊“高級”按鈕,選中“檢查數(shù)學(xué)運算溢出”選項,如下示意圖

總結(jié):
checked和unchecked是兩個不常用的關(guān)鍵字,但是他們倆是有用的,在需要的時候請記得用他們兩位,另外建議測試時開啟全局checked編譯器選項,謝謝。
1. 一段編譯沒通過的代碼
復(fù)制代碼 代碼如下:
int a = int.MaxValue * 2;
以上代碼段編譯沒有通過,在VS2010中會有一條紅色的波浪線指出這段代碼有問題:”The operation overflows at compile time in checked mode”。這說明了編譯器會在編譯時檢查數(shù)學(xué)運算是否溢出。但是編譯時能檢查出溢出的情況僅限于使用常量的運算。2中的代碼編譯器就不報不出錯誤來了。
2. 一段編譯通過但是不能得到正確結(jié)果的代碼
復(fù)制代碼 代碼如下:
int temp = int.MaxValue;
int a = temp * 2;
Console.Write(a);
我先把常量int.MaxValue的值給了臨時變量temp,然后使用臨時變量乘以2計算結(jié)果賦值給a;這段代碼是可以正常執(zhí)行的,執(zhí)行結(jié)果將輸出 -2。
這說明在運行時默認情況程序是不會檢查算術(shù)運算是否溢出的,cpu只管算,對于它來講按規(guī)則算就是了,結(jié)果對不對不是他的錯。
正常執(zhí)行了,而結(jié)果是錯誤的,這是非常危險的情況,該如何避免這種危險呢?請看3
3. 使用checked關(guān)鍵字,溢出時報警
復(fù)制代碼 代碼如下:
int temp = int.MaxValue;
try
{
int a = checked(temp * 2);
Console.WriteLine(a);
}
catch (OverflowException)
{
Console.WriteLine("溢出了,要處理喲");
}
使用checked關(guān)鍵字修飾temp*2的計算結(jié)果,并使用try catch在發(fā)生溢出時做處理。以上代碼將輸出:“溢出了,要處理喲”
問題是如果一段代碼中有很多算術(shù)運算都需要做溢出檢查,那會有很多checked修飾的表達式,怎么辦呢?請看4
4. checked關(guān)鍵字可以修飾一個語句塊,請看下面代碼
復(fù)制代碼 代碼如下:
int temp = int.MaxValue;
try
{
checked
{
int num = temp / 20;
int a = temp * 2;
int c = temp * 1000;
}
}
catch (OverflowException)
{
Console.WriteLine("溢出了,要處理喲");
}
以上程序輸出結(jié)果和3一樣
5. checked在避免算術(shù)溢出方面很有用,那么unchecked呢,它有用嗎?答案是肯定的,有時候我們不需要準確的計算結(jié)果,我們只是需要那么一個數(shù)而已,至于溢出不溢出的關(guān)系不大,比如說生成一個對象的HashCode,比如說根據(jù)一個算法計算出一個相對隨機數(shù),這都是不需要準確結(jié)果的。如下代碼片段
復(fù)制代碼 代碼如下:
class Person
{
public string Name { get; set; }
public string Title { get; set; }
public override int GetHashCode()
{
return unchecked(Name.GetHashCode() + Title.GetHashCode());
}
}
unchecked也可以修飾語句塊,其用法和checked完全一樣。
6. checked和unchecked是可以嵌套使用的,雖然沒啥意義。語句是否是checked以最近嵌套的checked或者unchecked決定
7. 從IL中看checked關(guān)鍵字
C#代碼:
復(fù)制代碼 代碼如下:
static void Main(string[] args)
{
int a = int.MaxValue;
int b = a * 2;
int c = checked(a * 2);
int d = unchecked(a + 3);
Console.Read();
}
對應(yīng)IL
復(fù)制代碼 代碼如下:
.method private hidebysig static void Main(string[] args) cil managed
{
.entrypoint
// Code size 26 (0x1a)
.maxstack 2
.locals init ([0] int32 a,
[1] int32 b,
[2] int32 c,
[3] int32 d)
IL_0000: nop
IL_0001: ldc.i4 0x7fffffff
IL_0006: stloc.0
IL_0007: ldloc.0
IL_0008: ldc.i4.2
IL_0009: mul
IL_000a: stloc.1
IL_000b: ldloc.0
IL_000c: ldc.i4.2
IL_000d: mul.ovf
IL_000e: stloc.2
IL_000f: ldloc.0
IL_0010: ldc.i4.3
IL_0011: add
IL_0012: stloc.3
IL_0013: call int32 [mscorlib]System.Console::Read()
IL_0018: pop
IL_0019: ret
} // end of method Program::Main
請看IL中的紅色和綠色加重顯示代碼,可以看出使用checked時,IL的運算是mul.ovf;不使用checked或者使用unchecked時的IL運算函數(shù)是mul或者add,不帶.ovf。
8. checked或者unchecked只影響其包圍的語句,不會影響到包圍的語句內(nèi)調(diào)用函數(shù)的代碼塊,如下示例:
復(fù)制代碼 代碼如下:
static void Main(string[] args)
{
int a = int.MaxValue;
int b = 20;
checked
{
int c = TestMethod(a, b);
Console.WriteLine(c);
}
}
static int TestMethod(int a, int b)
{
return a * b;
}
上面代碼將會正常執(zhí)行,checked語句塊并未起到應(yīng)有的作用。
9. 全局開啟或者關(guān)閉checked編譯選項
在項目屬性頁上選擇“生成”選項卡,然后點擊“高級”按鈕,選中“檢查數(shù)學(xué)運算溢出”選項,如下示意圖

總結(jié):
checked和unchecked是兩個不常用的關(guān)鍵字,但是他們倆是有用的,在需要的時候請記得用他們兩位,另外建議測試時開啟全局checked編譯器選項,謝謝。
相關(guān)文章
Unity3D Shader實現(xiàn)動態(tài)星空
這篇文章主要為大家詳細介紹了Unity3D Shader實現(xiàn)動態(tài)星空,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2020-04-04
C#中StringBuilder用法以及和String的區(qū)別分析
當(dāng)我們在初學(xué)使用C#時,常常會不知道該用StringBuilder合適還是用String高效,下面是我在學(xué)習(xí)當(dāng)中對StringBuilder和String的區(qū)別總結(jié),分享給大家。2013-03-03
C#結(jié)合JS實現(xiàn)HtmlTable動態(tài)添加行并保存到數(shù)據(jù)庫的流程步驟
在 Web 應(yīng)用項目中,實現(xiàn)一對多錄入的數(shù)據(jù)管理功能是一項常見的應(yīng)用,因此可以實現(xiàn)一個相對輕量化的設(shè)計實現(xiàn)表格的錄入,為保證功能的可用性、界面友好性,本文給大家介紹了C#結(jié)合JS實現(xiàn)HtmlTable動態(tài)添加行并保存到數(shù)據(jù)庫,需要的朋友可以參考下2024-12-12
vista和win7在windows服務(wù)中交互桌面權(quán)限問題解決方法:穿透Session 0 隔離
服務(wù)(Service)對于大家來說一定不會陌生,它是Windows 操作系統(tǒng)重要的組成部分。我們可以把服務(wù)想像成一種特殊的應(yīng)用程序,它隨系統(tǒng)的“開啟~關(guān)閉”而“開始~停止”其工作內(nèi)容,在這期間無需任何用戶參與2016-04-04
C#窗口轉(zhuǎn)向方式(由一個窗口,跳轉(zhuǎn)到另一個窗口)
這篇文章主要介紹了C#窗口轉(zhuǎn)向方式(由一個窗口,跳轉(zhuǎn)到另一個窗口)問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-07-07

