深入理解c# checked unchecked 關(guān)鍵字
更新時(shí)間:2013年05月18日 16:13:07 作者:
本篇文章是對(duì)c#中的checked unchecked 關(guān)鍵字進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
checked 和 unchecked關(guān)鍵字用來(lái)限定檢查或者不檢查數(shù)學(xué)運(yùn)算溢出的;如果使用了checked發(fā)生數(shù)學(xué)運(yùn)算溢出時(shí)會(huì)拋出OverflowException;如果使用了unchecked則不會(huì)檢查溢出,算錯(cuò)了也不會(huì)報(bào)錯(cuò)。
1. 一段編譯沒(méi)通過(guò)的代碼
int a = int.MaxValue * 2;
以上代碼段編譯沒(méi)有通過(guò),在VS2010中會(huì)有一條紅色的波浪線指出這段代碼有問(wèn)題:”The operation overflows at compile time in checked mode”。這說(shuō)明了編譯器會(huì)在編譯時(shí)檢查數(shù)學(xué)運(yùn)算是否溢出。但是編譯時(shí)能檢查出溢出的情況僅限于使用常量的運(yùn)算。2中的代碼編譯器就不報(bào)不出錯(cuò)誤來(lái)了。
2. 一段編譯通過(guò)但是不能得到正確結(jié)果的代碼
int temp = int.MaxValue;
int a = temp * 2;
Console.Write(a);
我先把常量int.MaxValue的值給了臨時(shí)變量temp,然后使用臨時(shí)變量乘以2計(jì)算結(jié)果賦值給a;這段代碼是可以正常執(zhí)行的,執(zhí)行結(jié)果將輸出 -2。
這說(shuō)明在運(yùn)行時(shí)默認(rèn)情況程序是不會(huì)檢查算術(shù)運(yùn)算是否溢出的,cpu只管算,對(duì)于它來(lái)講按規(guī)則算就是了,結(jié)果對(duì)不對(duì)不是他的錯(cuò)。
正常執(zhí)行了,而結(jié)果是錯(cuò)誤的,這是非常危險(xiǎn)的情況,該如何避免這種危險(xiǎn)呢?請(qǐng)看3
3. 使用checked關(guān)鍵字,溢出時(shí)報(bào)警
int temp = int.MaxValue;
try
{
int a = checked(temp * 2);
Console.WriteLine(a);
}
catch (OverflowException)
{
Console.WriteLine("溢出了,要處理喲");
}
使用checked關(guān)鍵字修飾temp*2的計(jì)算結(jié)果,并使用try catch在發(fā)生溢出時(shí)做處理。以上代碼將輸出:“溢出了,要處理喲”
問(wèn)題是如果一段代碼中有很多算術(shù)運(yùn)算都需要做溢出檢查,那會(huì)有很多checked修飾的表達(dá)式,怎么辦呢?請(qǐng)看4
4. checked關(guān)鍵字可以修飾一個(gè)語(yǔ)句塊,請(qǐng)看下面代碼
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呢,它有用嗎?答案是肯定的,有時(shí)候我們不需要準(zhǔn)確的計(jì)算結(jié)果,我們只是需要那么一個(gè)數(shù)而已,至于溢出不溢出的關(guān)系不大,比如說(shuō)生成一個(gè)對(duì)象的HashCode,比如說(shuō)根據(jù)一個(gè)算法計(jì)算出一個(gè)相對(duì)隨機(jī)數(shù),這都是不需要準(zhǔn)確結(jié)果的。如下代碼片段
class Person
{
public string Name { get; set; }
public string Title { get; set; }
public override int GetHashCode()
{
return unchecked(Name.GetHashCode() + Title.GetHashCode());
}
}
unchecked也可以修飾語(yǔ)句塊,其用法和checked完全一樣。
6. checked和unchecked是可以嵌套使用的,雖然沒(méi)啥意義。語(yǔ)句是否是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();
}
對(duì)應(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
請(qǐng)看IL中的紅色和綠色加重顯示代碼,可以看出使用checked時(shí),IL的運(yùn)算是mul.ovf;不使用checked或者使用unchecked時(shí)的IL運(yùn)算函數(shù)是mul或者add,不帶.ovf。
8. checked或者unchecked只影響其包圍的語(yǔ)句,不會(huì)影響到包圍的語(yǔ)句內(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;
}
上面代碼將會(huì)正常執(zhí)行,checked語(yǔ)句塊并未起到應(yīng)有的作用。
9. 全局開啟或者關(guān)閉checked編譯選項(xiàng)
在項(xiàng)目屬性頁(yè)上選擇“生成”選項(xiàng)卡,然后點(diǎn)擊“高級(jí)”按鈕,選中“檢查數(shù)學(xué)運(yùn)算溢出”選項(xiàng),如下示意圖

總結(jié):
checked和unchecked是兩個(gè)不常用的關(guān)鍵字,但是他們倆是有用的,在需要的時(shí)候請(qǐng)記得用他們兩位,另外建議測(cè)試時(shí)開啟全局checked編譯器選項(xiàng),謝謝。
1. 一段編譯沒(méi)通過(guò)的代碼
復(fù)制代碼 代碼如下:
int a = int.MaxValue * 2;
以上代碼段編譯沒(méi)有通過(guò),在VS2010中會(huì)有一條紅色的波浪線指出這段代碼有問(wèn)題:”The operation overflows at compile time in checked mode”。這說(shuō)明了編譯器會(huì)在編譯時(shí)檢查數(shù)學(xué)運(yùn)算是否溢出。但是編譯時(shí)能檢查出溢出的情況僅限于使用常量的運(yùn)算。2中的代碼編譯器就不報(bào)不出錯(cuò)誤來(lái)了。
2. 一段編譯通過(guò)但是不能得到正確結(jié)果的代碼
復(fù)制代碼 代碼如下:
int temp = int.MaxValue;
int a = temp * 2;
Console.Write(a);
我先把常量int.MaxValue的值給了臨時(shí)變量temp,然后使用臨時(shí)變量乘以2計(jì)算結(jié)果賦值給a;這段代碼是可以正常執(zhí)行的,執(zhí)行結(jié)果將輸出 -2。
這說(shuō)明在運(yùn)行時(shí)默認(rèn)情況程序是不會(huì)檢查算術(shù)運(yùn)算是否溢出的,cpu只管算,對(duì)于它來(lái)講按規(guī)則算就是了,結(jié)果對(duì)不對(duì)不是他的錯(cuò)。
正常執(zhí)行了,而結(jié)果是錯(cuò)誤的,這是非常危險(xiǎn)的情況,該如何避免這種危險(xiǎn)呢?請(qǐng)看3
3. 使用checked關(guān)鍵字,溢出時(shí)報(bào)警
復(fù)制代碼 代碼如下:
int temp = int.MaxValue;
try
{
int a = checked(temp * 2);
Console.WriteLine(a);
}
catch (OverflowException)
{
Console.WriteLine("溢出了,要處理喲");
}
使用checked關(guān)鍵字修飾temp*2的計(jì)算結(jié)果,并使用try catch在發(fā)生溢出時(shí)做處理。以上代碼將輸出:“溢出了,要處理喲”
問(wèn)題是如果一段代碼中有很多算術(shù)運(yùn)算都需要做溢出檢查,那會(huì)有很多checked修飾的表達(dá)式,怎么辦呢?請(qǐng)看4
4. checked關(guān)鍵字可以修飾一個(gè)語(yǔ)句塊,請(qǐng)看下面代碼
復(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呢,它有用嗎?答案是肯定的,有時(shí)候我們不需要準(zhǔn)確的計(jì)算結(jié)果,我們只是需要那么一個(gè)數(shù)而已,至于溢出不溢出的關(guān)系不大,比如說(shuō)生成一個(gè)對(duì)象的HashCode,比如說(shuō)根據(jù)一個(gè)算法計(jì)算出一個(gè)相對(duì)隨機(jī)數(shù),這都是不需要準(zhǔn)確結(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也可以修飾語(yǔ)句塊,其用法和checked完全一樣。
6. checked和unchecked是可以嵌套使用的,雖然沒(méi)啥意義。語(yǔ)句是否是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();
}
對(duì)應(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
請(qǐng)看IL中的紅色和綠色加重顯示代碼,可以看出使用checked時(shí),IL的運(yùn)算是mul.ovf;不使用checked或者使用unchecked時(shí)的IL運(yùn)算函數(shù)是mul或者add,不帶.ovf。
8. checked或者unchecked只影響其包圍的語(yǔ)句,不會(huì)影響到包圍的語(yǔ)句內(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;
}
上面代碼將會(huì)正常執(zhí)行,checked語(yǔ)句塊并未起到應(yīng)有的作用。
9. 全局開啟或者關(guān)閉checked編譯選項(xiàng)
在項(xiàng)目屬性頁(yè)上選擇“生成”選項(xiàng)卡,然后點(diǎn)擊“高級(jí)”按鈕,選中“檢查數(shù)學(xué)運(yùn)算溢出”選項(xiàng),如下示意圖

總結(jié):
checked和unchecked是兩個(gè)不常用的關(guān)鍵字,但是他們倆是有用的,在需要的時(shí)候請(qǐng)記得用他們兩位,另外建議測(cè)試時(shí)開啟全局checked編譯器選項(xiàng),謝謝。
相關(guān)文章
C#下載網(wǎng)頁(yè)并在控制臺(tái)輸出的方法
這篇文章主要介紹了C#下載網(wǎng)頁(yè)并在控制臺(tái)輸出的方法,涉及C#基于http協(xié)議進(jìn)行網(wǎng)頁(yè)抓取及控制臺(tái)輸出顯示的相關(guān)技巧,需要的朋友可以參考下2015-07-07Unity性能優(yōu)化Shader函數(shù)ShaderUtil.GetShaderGlobalKeywords用法示例
這篇文章主要為大家介紹了Unity性能優(yōu)化Shader函數(shù)ShaderUtil.GetShaderGlobalKeywords用法示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-09-09WPF實(shí)現(xiàn)圓形進(jìn)度條的示例代碼
這篇文章主要為大家詳細(xì)介紹了WPF如何實(shí)現(xiàn)圓形的進(jìn)度條,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)或工作有一定幫助,感興趣的小伙伴可以了解一下2023-01-01C#模擬實(shí)現(xiàn)鼠標(biāo)自動(dòng)點(diǎn)擊與消息發(fā)送功能
這篇文章主要為大家詳細(xì)介紹了C#如何利用windows api來(lái)模擬實(shí)現(xiàn)鼠標(biāo)點(diǎn)擊、右擊、雙擊以及發(fā)送文本功能,文中的示例代碼講解詳細(xì),感興趣的可以了解一下2022-08-08HashTable、HashSet和Dictionary的區(qū)別點(diǎn)總結(jié)
在本篇文章里小編給大家整理的是關(guān)于HashTable、HashSet和Dictionary的區(qū)別點(diǎn),需要的朋友們可以學(xué)習(xí)下。2020-03-03C#使用CEFSharp獲取動(dòng)態(tài)網(wǎng)頁(yè)源碼的演示步驟
CEFSharp是一個(gè)用C#編寫的庫(kù),它是Chromium Embedded Framework (CEF) 的.NET封裝和擴(kuò)展,CEF允許開發(fā)者在自己的應(yīng)用程序中嵌入一個(gè)功能強(qiáng)大的HTML渲染引擎,從而能夠呈現(xiàn)網(wǎng)頁(yè)內(nèi)容,本文介紹了C#如何使用CEFSharp獲取動(dòng)態(tài)網(wǎng)頁(yè)源碼,需要的朋友可以參考下2024-08-08深入解析C#設(shè)計(jì)模式編程中對(duì)建造者模式的運(yùn)用
這篇文章主要介紹了C#設(shè)計(jì)模式編程中對(duì)建造者模式的運(yùn)用,文中還介紹了在.NET框架下建造者模式編寫思路的實(shí)現(xiàn),需要的朋友可以參考下2016-02-02