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