
前言
大家都知道,C# 中可以用 using 關(guān)鍵字來簡化非托管資源(如文件流、數(shù)據(jù)庫連接等)的釋放,當(dāng)變量離開 using 作用的范圍后,會自動調(diào)用對象的 Dispose 方法,從而完成非托管資源的釋放。在 C#8.0,進(jìn)一步引入了簡化版的 "using聲明" 語法來避免多個 using 語句的嵌套,保證代碼的優(yōu)美,例如:
string connStr = "......";
using var conn = new SqlConnection(connStr);
conn.Open();
using var cmd = conn.CreateCommand();
cmd.CommandText = "select * from testdb";
using var reader = cmd.ExecuteReader();
while (reader.Read())
{
// ......
}
雖然 using
語句非常有用,但在實(shí)際使用過程中也存在一些潛在的問題,不可不察!
可能的陷阱
嵌套使用 using
語句
當(dāng)多個 using
語句嵌套在一起,內(nèi)部 using
語句中的資源在釋放時,可能會把外部 using
語句中的資源也釋放掉,比如:
using (Stream stream = new FileStream("d:\1.txt", FileMode.OpenOrCreate))
{
using (StreamWriter writer = new StreamWriter(stream))
{
// ......
}
}
例子中當(dāng)內(nèi)部的 writer
釋放時,會同時釋放外部的 stream
對象,這是因?yàn)?StreamWriter 類型 Dispose 機(jī)制所造成,所以當(dāng)我們不太清除內(nèi)層對象是否與外層對象有關(guān)系時,尤其要慎用 using
語句。
除此之外,外部的 using
有時候也可能會在內(nèi)部的 using
結(jié)束前就釋放資源,導(dǎo)致意外的問題發(fā)生,比如數(shù)據(jù)庫連接。
資源釋放順序:
當(dāng)多個 using
語句嵌套在一起時,如果多個資源需要按照特定順序釋放時,using
語句可能無法保證這一順序,導(dǎo)致意外的問題發(fā)生。
作用域:
簡化版的 "using聲明" 語法的作用域是整個方法體,所以很容易導(dǎo)致意外的問題發(fā)生,比如以下代碼:
void usingTest()
{
using var outStream = File.OpenWrite("d:/1.txt");
using var writer = new StreamWriter(outStream);
writer.WriteLine("Hello world");
string s = File.ReadAllText("d:/1.txt");
Console.WriteLine(s);
}
當(dāng)代碼執(zhí)行到下面這行代碼時,就會提示文件被占用的錯誤。
string s = File.ReadAllText("d:/1.txt");
總結(jié)
using
語句本質(zhì)上是 try-finally
的語法糖,所以當(dāng)多個 using
語句嵌套在一起的時候,實(shí)際上就是多個 try-finally
語句嵌套在一起,所以造成一些奇怪的問題也就不奇怪了,尤其是 C#8.0 進(jìn)一步簡化 using
語句之后。
我們在享受 using
語句帶來的便利的同時,也要注意 using
語句正確的使用場景,尤其是在需要使用嵌套 using
語句的時候,這樣才能提高程序的健壯性。
最后,using
語句除了用于釋放非托管資源之外,還在其它的用途,比如引用命名空間、為命名空間或類型創(chuàng)建別名等,有興趣的童鞋可以繼續(xù)深入了解。
該文章在 2024/12/13 9:25:16 編輯過