Swift - defer關鍵字(推遲執行)


在一些語言中,有try/finally這樣的控制語句,比如Java。

這種語句可以讓我們在finally代碼塊中執行必須要執行的代碼,不管之前怎樣的興風作浪。

在Swift 2.0中,Apple提供了defer關鍵字,讓我們可以實現同樣的效果。

func checkSomething() {
    
    print("CheckPoint 1")
    doSomething()
    print("CheckPoint 4")
    
}

func doSomething() {
    
    print("CheckPoint 2")
    defer {
        print("Clean up here")
    }
    print("CheckPoint 3")
    
}

checkSomething() // CheckPoint 1, CheckPoint 2, CheckPoint 3, Clean up here, CheckPoint 4

上述示例可以看到,在打印出“CheckPoint 2”之后並沒有打印出“Clean up here”,而是“CheckPoint 3”,這就是defer的作用,它對進行了print("Clean up here")延遲。我們再來看一個I/O的示例:

// 偽代碼
func writeSomething() {
    
    let file = OpenFile()
    
    let ioStatus = fetchIOStatus()
    guard ioStatus != "error" else {
        return
    }
    file.write()
    
    closeFile(file)
    
}

上述示例是一個I/O操作的偽代碼,如果獲取到的ioStatus正常,那么該方法沒有問題,

如果ioStatus取到的是error,那么會被guard語句抓到執行return操作,

這樣的話closeFile(file)就永遠都不會執行了,一個嚴重的Bug就這樣產生了。下面我們看看如何用defer來解決這個問題:

// 偽代碼
func writeSomething() {
    
    let file = OpenFile()
    defer {
        closeFile(file)
    }
    
    let ioStatus = fetchIOStatus()
    guard ioStatus != "error" else {
        return
    }
    file.write()
    
}

我們將closeFile(file)放在defer代碼塊里,這樣即使ioStatuserror,在執行return前會先執行defer里的代碼,這樣就保證了不管發生什么,最后都會將文件關閉。

需要注意的是, 雖然說defer的內容會在return之前執行, 但是如果defer定義在return之后, 那么還是不會執行defter的內容, 也就是說, defer關鍵字必須比return早出現。

放在return之后:

var str = "Hello, playground"

func show() {
    print("這里是即將要推遲(但一定會)執行的代碼")
}

func test() {
    
    if str.characters.count >= 2 {
        print("跳出該方法的執行")
        return
    }
    
    defer {
        show()
    }
}

test()  // 輸出結果: 跳出該方法的執行

放在return之前:

var str = "Hello, playground"

func show() {
    print("這里是即將要推遲(但一定會)執行的代碼")
}

func test() {
    
    defer {
        show()
    }
    
    if str.characters.count >= 2 {
        print("跳出該方法的執行")
        return
    }
}

test()  // 輸出結果: 跳出該方法的執行 這里是即將要推遲(但一定會)執行的代碼


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM