如何使用 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 块比等待垃圾回收更好。

另请参阅