C#using语句
我真的想把这个弄出来。 请看下面的代码:
using (DataTable resultTable = DBUtility.GetSingleDBTableResult(connectionString, "SELECT * FROM MyDBTable")) { List resultsList = new List(); foreach (DataRow dataRow in resultTable.Rows) { resultsList.Add(dataRow[0].ToString()); } return resultsList; }
数据表是否处理? 有人可以解释这是如何转换为try / catch / finally块的吗? MSDN声明如果发生exception,仍会调用Dispose方法但是return语句呢?
或者我应该使用下面的代码:
List resultsList = new List(); using (DataTable resultTable = DBUtility.GetSingleDBTableResult(connectionString, "SELECT * FROM MyDBTable")) { foreach (DataRow dataRow in resultTable.Rows) { resultsList.Add(dataRow[0].ToString()); } } return resultsList;
可能应该使用第二个,但我只需要启蒙:)。 请解释。 谢谢。
using
语句只是语法糖,它被转换为try / finally块。 从代码开始,这里是C#编译器如何将using
块转换为try / finally块。
try { DataTable resultTable = DBUtility.GetSingleDBTableResult(connectionString, "SELECT * FROM MyDBTable"); List resultsList = new List (); foreach (DataRow dataRow in resultTable.Rows) { resultsList.Add(dataRow[0].ToString()); } return resultsList; } finally { resultTable.Dispose(); }
正如您从代码中看到的那样,无论return语句如何,resultTable都会被放置。 using块仅确保对象在范围之后被释放。
您的第一个代码对我来说没问题,无需更改。
Using
不捕获exception,它只保证.Dispose()
调用。
这是因为,
using (ResourceType resource = new ResourceType())
相当于:
ResourceType resource; try { resource = new ResourceType(); /* The insides of the using block */ } finally { resource.Dispose(); }
始终会评估.Dispose()
调用。 如果您在using
块中返回(在“真正”返回之前),则甚至会评估Dispose
调用。 如果抛出exception,甚至会评估Dispose
调用。
但是,如果抛出exception,该exception仍将阻止对后续代码行进行求值(除了始终求值的.Dispose()
)。
因此,如果发生exception,您的return
将不会在您的任何一个语句中返回,但您的DataTable
仍将被处置。
如果您想保证发生返回,即使发生错误,您也希望执行以下操作:
List resultsList = new List(); try { using (DataTable resultTable = DBUtility.GetSingleDBTableResult(connectionString, "SELECT * FROM MyDBTable")) { foreach (DataRow dataRow in resultTable.Rows) { resultsList.Add(dataRow[0].ToString()); } } } catch { } return resultsList;
在两种情况下都处理DataTable( .Dispose
)。
它被翻译成了一个try/finally
,在finally
中调用了Dispose
。 最后,顾名思义,即使您调用return
也会调用它。
在两个示例中,都将调用Dispose
。 这是因为using语句扩展为try / finally块。
阅读C#语言规范 (8.13 using语句)以找出各种场景(对于引用类型,非可空值类型和动态类型)。
由于DataTable
是引用类型,因此您的第一个示例将扩展为以下内容:
{ DataTable resultTable = DBUtility.GetSingleDBTableResult(connectionString, "SELECT * FROM MyDBTable"); try { List resultsList = new List (); foreach (DataRow dataRow in resultTable.Rows) { resultsList.Add(dataRow[0].ToString()); } return resultsList; } finally { if (resultTable != null) ((IDisposable)resultTable).Dispose(); } }