如果在一个代码域中包含多个defer块,它们将按相反顺序执行;你可以把它们想象成一个栈。乍一看defer以相反顺序执行感觉很奇怪,但不管怎样,如果我们看一个实际的例子,就会觉得有道理:
guard let database = openDatabase(...) else { return }
defer { closeDatabase(database) }
guard let connection = openConnection(database) else { return } defer { closeConnection(connection) }
guard let result = runQuery(connection,...) else { return }
如果有错误发生 —– 举个栗子,在runQuery调用的时候 —– 我们首先需要关闭连接然后才是关闭数据库。因为defer自动以相反顺序执行。runQuery以来与openConnection,后者依赖于openDatabase。因此,清理资源需要以相反顺序执行。
这里也有些情况defer块不会执行:当你的程序发生段错误时,或者当期引起一个致命错误(比如使用fatalError或强制解包一个nil变量),所有执行将会被立即挂起。