亚洲最大看欧美片,亚洲图揄拍自拍另类图片,欧美精品v国产精品v呦,日本在线精品视频免费

  • 站長(zhǎng)資訊網(wǎng)
    最全最豐富的資訊網(wǎng)站

    Go語言中延遲執(zhí)行語句是什么

    在Go語言中,延遲執(zhí)行語句是defer語句,語法“defer 任意語句”。defer語句會(huì)將其后面跟隨的語句進(jìn)行延遲處理,在defer歸屬的函數(shù)即將返回時(shí),將延遲處理的語句按defer的逆序進(jìn)行執(zhí)行;也就是說,先被defer的語句最后被執(zhí)行,最后被defer的語句,最先被執(zhí)行。

    Go語言中延遲執(zhí)行語句是什么

    本教程操作環(huán)境:windows7系統(tǒng)、GO 1.18版本、Dell G3電腦。

    Go 語言中的延遲執(zhí)行語句(defer語句)

    Go語言中存在一種延遲執(zhí)行的語句,由defer關(guān)鍵字標(biāo)識(shí)。

    defer 關(guān)鍵字會(huì)將其后面跟隨的語句進(jìn)行延遲處理,在 defer 歸屬的函數(shù)即將返回時(shí),將延遲處理的語句按 defer 的逆序進(jìn)行執(zhí)行,也就是說,先被 defer 的語句最后被執(zhí)行,最后被 defer 的語句,最先被執(zhí)行。

    格式如下:

    defer 任意語句
    登錄后復(fù)制

    defer后的語句不會(huì)被馬上執(zhí)行,在defer所屬的函數(shù)即將返回時(shí),函數(shù)體中的所有defer語句將會(huì)按出現(xiàn)的順序被逆序執(zhí)行,即函數(shù)體中的最后一個(gè)defer語句最先被執(zhí)行。

    package main  import "fmt"  func main(){ 	fmt.Println("start now") 	defer fmt.Println("這是第一句defer語句") 	defer fmt.Println("這是第二句defer語句") 	defer fmt.Println("這是第三句defer語句") 	fmt.Println("end") }
    登錄后復(fù)制

    執(zhí)行結(jié)果如下:

    start now end 這是第三句defer語句 這是第二句defer語句 這是第一句defer語句
    登錄后復(fù)制

    由于defer語句是在當(dāng)前函數(shù)即將返回時(shí)被調(diào)用,所以defer常常被用來釋放資源。

    多個(gè)延遲執(zhí)行語句的處理順序

    當(dāng)有多個(gè) defer 行為被注冊(cè)時(shí),它們會(huì)以逆序執(zhí)行(類似棧,即后進(jìn)先出),下面的代碼是將一系列的數(shù)值打印語句按順序延遲處理,如下所示:

    package main import (     "fmt" ) func main() {     fmt.Println("defer begin")     // 將defer放入延遲調(diào)用棧     defer fmt.Println(1)     defer fmt.Println(2)     // 最后一個(gè)放入, 位于棧頂, 最先調(diào)用     defer fmt.Println(3)     fmt.Println("defer end") }
    登錄后復(fù)制

    代碼輸出如下:

    defer begin defer end 3 2 1
    登錄后復(fù)制

    結(jié)果分析如下:

    • 代碼的延遲順序與最終的執(zhí)行順序是反向的。

    • 延遲調(diào)用是在 defer 所在函數(shù)結(jié)束時(shí)進(jìn)行,函數(shù)結(jié)束可以是正常返回時(shí),也可以是發(fā)生宕機(jī)時(shí)。

    使用延遲執(zhí)行語句在函數(shù)退出時(shí)釋放資源

    處理業(yè)務(wù)或邏輯中涉及成對(duì)的操作是一件比較煩瑣的事情,比如打開和關(guān)閉文件、接收請(qǐng)求和回復(fù)請(qǐng)求、加鎖和解鎖等。在這些操作中,最容易忽略的就是在每個(gè)函數(shù)退出處正確地釋放和關(guān)閉資源。

    defer 語句正好是在函數(shù)退出時(shí)執(zhí)行的語句,所以使用 defer 能非常方便地處理資源釋放問題。

    1) 使用延遲并發(fā)解鎖

    在下面的例子中會(huì)在函數(shù)中并發(fā)使用 map,為防止競(jìng)態(tài)問題,使用 sync.Mutex 進(jìn)行加鎖,參見下面代碼:

    var (     // 一個(gè)演示用的映射     valueByKey      = make(map[string]int)     // 保證使用映射時(shí)的并發(fā)安全的互斥鎖     valueByKeyGuard sync.Mutex ) // 根據(jù)鍵讀取值 func readValue(key string) int {     // 對(duì)共享資源加鎖     valueByKeyGuard.Lock()     // 取值     v := valueByKey[key]     // 對(duì)共享資源解鎖     valueByKeyGuard.Unlock()     // 返回值     return v }
    登錄后復(fù)制

    代碼說明如下:

    • 第 3 行,實(shí)例化一個(gè) map,鍵是 string 類型,值為 int。

    • 第 5 行,map 默認(rèn)不是并發(fā)安全的,準(zhǔn)備一個(gè) sync.Mutex 互斥量保護(hù) map 的訪問。

    • 第 9 行,readValue() 函數(shù)給定一個(gè)鍵,從 map 中獲得值后返回,該函數(shù)會(huì)在并發(fā)環(huán)境中使用,需要保證并發(fā)安全。

    • 第 11 行,使用互斥量加鎖。

    • 第 13 行,從 map 中獲取值。

    • 第 15 行,使用互斥量解鎖。

    • 第 17 行,返回獲取到的 map 值。

    使用 defer 語句對(duì)上面的語句進(jìn)行簡(jiǎn)化,參考下面的代碼。

    func readValue(key string) int {     valueByKeyGuard.Lock()         // defer后面的語句不會(huì)馬上調(diào)用, 而是延遲到函數(shù)結(jié)束時(shí)調(diào)用     defer valueByKeyGuard.Unlock()     return valueByKey[key] }
    登錄后復(fù)制

    上面的代碼中第 6~8 行是對(duì)前面代碼的修改和添加的代碼,代碼說明如下:

    • 第 6 行在互斥量加鎖后,使用 defer 語句添加解鎖,該語句不會(huì)馬上執(zhí)行,而是等 readValue() 函數(shù)返回時(shí)才會(huì)被執(zhí)行。

    • 第 8 行,從 map 查詢值并返回的過程中,與不使用互斥量的寫法一樣,對(duì)比上面的代碼,這種寫法更簡(jiǎn)單。

    2) 使用延遲釋放文件句柄

    文件的操作需要經(jīng)過打開文件、獲取和操作文件資源、關(guān)閉資源幾個(gè)過程,如果在操作完畢后不關(guān)閉文件資源,進(jìn)程將一直無法釋放文件資源,在下面的例子中將實(shí)現(xiàn)根據(jù)文件名獲取文件大小的函數(shù),函數(shù)中需要打開文件、獲取文件大小和關(guān)閉文件等操作,由于每一步系統(tǒng)操作都需要進(jìn)行錯(cuò)誤處理,而每一步處理都會(huì)造成一次可能的退出,因此就需要在退出時(shí)釋放資源,而我們需要密切關(guān)注在函數(shù)退出處正確地釋放文件資源,參考下面的代碼:

    // 根據(jù)文件名查詢其大小 func fileSize(filename string) int64 {     // 根據(jù)文件名打開文件, 返回文件句柄和錯(cuò)誤     f, err := os.Open(filename)     // 如果打開時(shí)發(fā)生錯(cuò)誤, 返回文件大小為0     if err != nil {         return 0     }     // 取文件狀態(tài)信息     info, err := f.Stat()         // 如果獲取信息時(shí)發(fā)生錯(cuò)誤, 關(guān)閉文件并返回文件大小為0     if err != nil {         f.Close()         return 0     }     // 取文件大小     size := info.Size()     // 關(guān)閉文件     f.Close()         // 返回文件大小     return size }
    登錄后復(fù)制

    代碼說明如下:

    • 第 2 行,定義獲取文件大小的函數(shù),返回值是 64 位的文件大小值。

    • 第 5 行,使用 os 包提供的函數(shù) Open(),根據(jù)給定的文件名打開一個(gè)文件,并返回操作文件用的句柄和操作錯(cuò)誤。

    • 第 8 行,如果打開的過程中發(fā)生錯(cuò)誤,如文件沒找到、文件被占用等,將返回文件大小為 0。

    • 第 13 行,此時(shí)文件句柄 f 可以正常使用,使用 f 的方法 Stat() 來獲取文件的信息,獲取信息時(shí),可能也會(huì)發(fā)生錯(cuò)誤。

    • 第 16~19 行對(duì)錯(cuò)誤進(jìn)行處理,此時(shí)文件是正常打開的,為了釋放資源,必須要調(diào)用 f 的 Close() 方法來關(guān)閉文件,否則會(huì)發(fā)生資源泄露。

    • 第 22 行,獲取文件大小。

    • 第 25 行,關(guān)閉文件、釋放資源。

    • 第 28 行,返回獲取到的文件大小。

    在上面的例子中,第 25 行是對(duì)文件的關(guān)閉操作,下面使用 defer 對(duì)代碼進(jìn)行簡(jiǎn)化,代碼如下:

    func fileSize(filename string) int64 {     f, err := os.Open(filename)     if err != nil {         return 0     }     // 延遲調(diào)用Close, 此時(shí)Close不會(huì)被調(diào)用     defer f.Close()     info, err := f.Stat()     if err != nil {         // defer機(jī)制觸發(fā), 調(diào)用Close關(guān)閉文件         return 0     }     size := info.Size()     // defer機(jī)制觸發(fā), 調(diào)用Close關(guān)閉文件     return size }
    登錄后復(fù)制

    代碼中加粗部分為對(duì)比前面代碼而修改的部分,代碼說明如下:

    • 第 10 行,在文件正常打開后,使用 defer,將 f.Close() 延遲調(diào)用,注意,不能將這一句代碼放在第 4 行空行處,一旦文件打開錯(cuò)誤,f 將為空,在延遲語句觸發(fā)時(shí),將觸發(fā)宕機(jī)錯(cuò)誤。

    • 第 16 行和第 22 行,defer 后的語句(f.Close())將會(huì)在函數(shù)返回前被調(diào)用,自動(dòng)釋放資源。

    贊(0)
    分享到: 更多 (0)
    網(wǎng)站地圖   滬ICP備18035694號(hào)-2    滬公網(wǎng)安備31011702889846號(hào)