C#語言中也是如此。當(dāng)多個條件進行邏輯與操作的時候,判定會從表達式左邊執(zhí)行到右邊,遇到任何一個為假,后面就都不做了。這很聰明,然而如果后面的條件會拋出異常,就是個潛在的問題。一旦之前的條件為真,就會繼續(xù)執(zhí)行,執(zhí)行到拋出異常的條件時,程序就爆了,哈哈。
我們可以寫個簡單的demo試試。下面的這段代碼是坑爹的,之后我會說明原因,但大家可以先從直觀的層面上理解一下,最后我會給出正確的測試方法。
復(fù)制代碼 代碼如下:
static void Main(string[] args)
{
DataSet ds = null;
if (false ds.Tables[0].Rows.Count > 0)
{
Console.WriteLine("Fuck");
}
else
{
Console.WriteLine("Shit");
}
if (true ds.Tables[0].Rows.Count > 0)
{
Console.WriteLine("WOW");
}
else
{
Console.WriteLine("KAO");
}
Console.ReadKey();
}
這段代碼乍看沒問題,并且在運行時也給出了我們期望的結(jié)果,即第一段語句輸出Shit,不拋出異常(當(dāng)前面為false,后面會拋異常的ds.Tables[0].Rows.Count > 0就不做),而第二段語句因為之前是true,所以要執(zhí)行對dataset的判斷,所以拋出異常。但如果用reflector反編譯程序集,就會發(fā)現(xiàn),編譯器已經(jīng)把上面的代碼優(yōu)化成了下面這種形式,我們的if語句中寫死的true和false已經(jīng)被閹割掉了,所以并不能說明if語句執(zhí)行的問題。
復(fù)制代碼 代碼如下:
private static void Main(string[] args)
{
DataSet ds = null;
Console.WriteLine("Shit");
if (ds.Tables[0].Rows.Count > 0)
{
Console.WriteLine("WOW");
}
else
{
Console.WriteLine("KAO");
}
Console.ReadKey();
}
其實,如果你仔細(xì)觀察,在輸入這段代碼的過程中,VS就已經(jīng)提示if (false ds.Tables[0].Rows.Count > 0)中,后者是不可達的。這是即時編譯的效果。既然即時編譯說后面的代碼不可達,就意味著不可達的代碼會在編譯期被切掉。因此,剛才我們在上面看到的編譯結(jié)果也就是自然的事情了。
同樣,如果你直接把1 == 0, 1 == 1這樣的條件拼上去的話,編譯器也會發(fā)現(xiàn)的。所以我們要找一種不會被編譯器發(fā)現(xiàn)的寫法,要讓我們的條件判定代碼只能在運行時執(zhí)行,而不是編譯時被調(diào)整。比如下面這種:
復(fù)制代碼 代碼如下:
static void Main(string[] args)
{
DataSet ds = null;
int i = 0;
int j = 1;
if (i + j == 0 ds.Tables[0].Rows.Count > 0)
{
Console.WriteLine("Fuck");
}
else
{
Console.WriteLine("Shit");
}
if (i + j == 1 ds.Tables[0].Rows.Count > 0)
{
Console.WriteLine("WOW");
}
else
{
Console.WriteLine("KAO");
}
Console.ReadKey();
}
我們再來執(zhí)行,發(fā)現(xiàn)這次的結(jié)果是真正意義的滿足了我們的目的,說明了當(dāng)多個條件進行邏輯與的時候,C#的執(zhí)行機制:
寫這篇文章的意義,是為了讓大家在寫程序的時候,注意條件中可能發(fā)生異常的地方。比如我們模擬String.IsNullOrEmpty()。
在or關(guān)系中,只要有一個true,整個表達式就是true了。但如果你讓可能引發(fā)異常的語句先于之后會返回true的語句執(zhí)行,就會爆。
比如這樣寫的話會爆,因為判斷Length的前提是得有個string:
復(fù)制代碼 代碼如下:
public static bool IsNullOrEmpty(string str)
{
if (str.Length == 0 || str == null)
{
return true;
}
return false;
}
這樣寫就正常:
復(fù)制代碼 代碼如下:
public static bool IsNullOrEmpty(string str)
{
if (str == null || str.Length == 0)
{
return true;
}
return false;
}
微軟是這樣寫的,碉堡了!
復(fù)制代碼 代碼如下:
public static bool IsNullOrEmpty(string value)
{
if (value != null)
{
return (value.Length == 0);
}
return true;
}
上面這段代碼可以用reflector打開mscorlib中的System.String找到~
您可能感興趣的文章:- C#條件語句、循環(huán)語句(if、while)
- C#中if語句使用概述
- c#入門之分支語句使用方法(三元運算符、if語句、switch語句)
- 利用C#9.0新語法如何提升if語句美感