如何使用 finally 执行清理代码
finally
语句的用途是确保立即进行对象(通常是容纳外部资源的对象)的必要清理(即使引发异常)。 这类清理的一个示例是在使用之后立即对 FileStream 调用 Close(而不是等待公共语言运行时对对象进行垃圾回收),如下所示:
static void CodeWithoutCleanup()
{
FileStream? file = null;
FileInfo fileInfo = new FileInfo("./file.txt");
file = fileInfo.OpenWrite();
file.WriteByte(0xF);
file.Close();
}
示例
若要将上面的代码转换为 try-catch-finally
语句,可将清理代码与工作代码分开,如下所示。
static void CodeWithCleanup()
{
FileStream? file = null;
FileInfo? fileInfo = null;
try
{
fileInfo = new FileInfo("./file.txt");
file = fileInfo.OpenWrite();
file.WriteByte(0xF);
}
catch (UnauthorizedAccessException e)
{
Console.WriteLine(e.Message);
}
finally
{
file?.Close();
}
}
因为可能会在 try
块中进行 OpenWrite()
调用之前的任何时候发生异常,或 OpenWrite()
调用本身可能会失败,所以我们不保证在尝试关闭文件时文件处于打开状态。 finally
块添加了一个检查,以便在调用 Close 方法之前确保 FileStream 对象不是 null
。 如果不进行 null
检查,则 finally
块可能会引发自己的 NullReferenceException,但是在可能时应避免在 finally
块中引发异常。
数据库连接是在 finally
块中进行关闭的另一个很好的候选项。 因为与数据库服务器之间的允许连接数有时会受到限制,所以应尽快关闭数据库连接。 如果在可以关闭连接之前引发异常,则使用 finally
块比等待垃圾回收更好。