亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

golang中的defer函數(shù)理解

 更新時(shí)間:2022年10月24日 16:11:58   作者:西京刀客  
defer是Go語(yǔ)言中的延遲執(zhí)行語(yǔ)句,用來添加函數(shù)結(jié)束時(shí)執(zhí)行的代碼,常用于釋放某些已分配的資源、關(guān)閉數(shù)據(jù)庫(kù)連接、斷開socket連接、解鎖一個(gè)加鎖的資源,這篇文章主要介紹了golang中的defer函數(shù)理解,需要的朋友可以參考下

golang的defer

什么是defer

defer的的官方文檔:https://golang.org/ref/spec#Defer_statements

go語(yǔ)言中defer可以完成延遲功能,當(dāng)前函數(shù)執(zhí)行完成后再執(zhí)行defer的代碼塊。通過defer,我們可以在代碼中優(yōu)雅的關(guān)閉/清理代碼中所使用的變量。

defer是Go語(yǔ)言中的延遲執(zhí)行語(yǔ)句,用來添加函數(shù)結(jié)束時(shí)執(zhí)行的代碼,常用于釋放某些已分配的資源、關(guān)閉數(shù)據(jù)庫(kù)連接、斷開socket連接、解鎖一個(gè)加鎖的資源。

Go語(yǔ)言機(jī)制擔(dān)保一定會(huì)執(zhí)行defer語(yǔ)句中的代碼。其它語(yǔ)言中也有類似的機(jī)制,比如Java、C#語(yǔ)言里的finally語(yǔ)句,C++語(yǔ)言里的析構(gòu)函數(shù)(Destructor)可以起類似的作用,C++語(yǔ)言機(jī)制擔(dān)保在對(duì)象被銷毀前一定會(huì)執(zhí)行析構(gòu)函數(shù)中的代碼。C++中的析構(gòu)函數(shù)析構(gòu)的是對(duì)象,Go中的defer析構(gòu)的是函數(shù)。

理解defer

defer什么時(shí)間執(zhí)行(defer、 return、返回值 三者的執(zhí)行順序)

defer只有在當(dāng)前函數(shù)執(zhí)行完畢后,才會(huì)執(zhí)行。描述其實(shí)不太精確

go中的return語(yǔ)句并不是原子性操作,一般是分為兩步:

  • 將返回值賦值給一個(gè)變量
  • 執(zhí)行RET指令

return并不是原子性操作,是通過一個(gè)變量賦值和ret指令來完成的。defer就執(zhí)行在1之后,2之前。defer的執(zhí)行順序在return之后,但是在返回值返回給調(diào)用方之前,所以使用defer可以達(dá)到修改返回值的目的。

defer、 return、返回值 三者的執(zhí)行順序是 : return 最先給返回值賦值;接著 defer 開始執(zhí)行一些收尾工作;最后 RET 指令攜帶返回值退出函數(shù)。

package main

import (
    "fmt"
)

func main() {
    ret := test()
    fmt.Println("test return:", ret)
}
// func test() ( int) {  這種就是匿名返回值
//返回值改為命名返回值, 具名返回值。即返回值帶有名字, 這樣我們?cè)趫?zhí)行defer的時(shí)候相當(dāng)于修改了返回值的值
func test() (i int) {
    //var i int

    defer func() {
        i++
        fmt.Println("test defer, i = ", i)
    }()

    return i
}

注意: 這塊驗(yàn)證使用了具名返回值 func test() (i int) { 中的(i int) 。 測(cè)試結(jié)果滿足我們預(yù)期。

編碼中,我們要特別注意, go語(yǔ)言中匿名返回值和命名返回值對(duì)defer的影響。不過一般我們都是使用命名返回值。

一個(gè)主函數(shù)擁有一個(gè)匿名的返回值,返回時(shí)使用字面值,比如返回”1”、”2”、”Hello”這樣的值,這種情況下defer語(yǔ)句是無法操作返回值的。

defer輸出的值,就是定義時(shí)的值。而不是defer真正執(zhí)行時(shí)的變量值(注意引用情況)

defer函數(shù)會(huì)在return之后被調(diào)用。那么這段函數(shù)執(zhí)行完之后,是不用應(yīng)該輸出1呢?

package main

import "fmt"

func test1() {
	i := 0
	defer fmt.Println(i)
	i++
	return
}

func main() {
	test1()
}

輸出結(jié)果:0

雖然我們?cè)赿efer后面定義的是一個(gè)帶變量的函數(shù): fmt.Println(i). 但這個(gè)變量(i)在defer被聲明的時(shí)候,就已經(jīng)確定其確定的值了。

總結(jié): 因?yàn)閐efer后面的函數(shù)在入棧的時(shí)候保存的是入棧那一刻的值,而當(dāng)時(shí)i的值是0,所以后期對(duì)i修改,并不會(huì)影響棧內(nèi)函數(shù)的值。

我們?cè)倏聪乱粋€(gè)例子:

package main

import "fmt"


func test2() {
	x := 10
	defer func(a *int) {
		fmt.Println(*a)
	}(&x)
	x++
}

func main() {
	test2()
}

輸出結(jié)果: 11

這里為什么和前面結(jié)論不一樣呢?
這里defer后面函數(shù)入棧的時(shí)候存入的執(zhí)行變量x的指針。所以,后期x值改變的時(shí)候,輸出結(jié)果也會(huì)改變。

總結(jié): 需要注意引用情況。對(duì)于指針類型參數(shù),規(guī)則仍然適用,只不過延遲函數(shù)的參數(shù)是一個(gè)地址值,這種情況下,defer后面的語(yǔ)句對(duì)變量的修改會(huì)影響延遲函數(shù)。

多個(gè)defer,執(zhí)行順序

package main

import (
    "fmt"
)

func main() {
    defer fmt.Println("main defer1")
    test()
    defer fmt.Println("main defer2")
}

func test() () {
    defer func() {
        fmt.Println("test defer1")
    }()
    defer func() {
        fmt.Println("test defer2")
    }()
}

輸出結(jié)果:

test defer2
test defer1
main defer2
main defer1

總結(jié): 后進(jìn)先出(LIFO)的順序執(zhí)行,即先出現(xiàn)的 defer 最后執(zhí)行。即:多個(gè)defer語(yǔ)句的執(zhí)行順序是逆序執(zhí)行。

defer的函數(shù)一定會(huì)執(zhí)行么?

defer是Go語(yǔ)言中的延遲執(zhí)行語(yǔ)句,用來添加函數(shù)結(jié)束時(shí)執(zhí)行的代碼,常用于釋放某些已分配的資源、關(guān)閉數(shù)據(jù)庫(kù)連接、斷開socket連接、解鎖一個(gè)加鎖的資源。

Go語(yǔ)言機(jī)制擔(dān)保一定會(huì)執(zhí)行defer語(yǔ)句中的代碼。其它語(yǔ)言中也有類似的機(jī)制,比如Java、C#語(yǔ)言里的finally語(yǔ)句,C++語(yǔ)言里的析構(gòu)函數(shù)(Destructor)可以起類似的作用,C++語(yǔ)言機(jī)制擔(dān)保在對(duì)象被銷毀前一定會(huì)執(zhí)行析構(gòu)函數(shù)中的代碼。C++中的析構(gòu)函數(shù)析構(gòu)的是對(duì)象,Go中的defer析構(gòu)的是函數(shù)。

panic情況

網(wǎng)上demo:

package main

import (
	"fmt"
)

func test1() {
	fmt.Println("test")
}

func test2() {
	panic(1)
}
func main() {
	fmt.Println("main start")
	defer test1()
	test2() //造panic
	fmt.Println("main end")
}

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

main start
test                                                                  
panic: 1                                                                                                                                   
goroutine 1 [running]:                                                
main.test2(...)         

總結(jié): 我們發(fā)現(xiàn)正常的panic,還是會(huì)調(diào)我們的defer的,并且在會(huì)在panic之前執(zhí)行。

os.Exit情況

網(wǎng)上demo:

package main

import (
	"fmt"
	"os"
)

func test1() {
	fmt.Println("test")
}

func main() {
	fmt.Println("main start")
	defer test1()
	fmt.Println("main end")
	os.Exit(0)
}

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

main start
main end

總結(jié): 如果在當(dāng)前函數(shù)里是因?yàn)閳?zhí)行了os.Exit退出,而不是正常return退出或者panic退出,那程序會(huì)立即停止,被defer的函數(shù)調(diào)用不會(huì)執(zhí)行。

kill情況(Ctrl+C)

package main

import (
	"fmt"
	"time"
)

func test1() {
	fmt.Println("test")
}

func test2() {
	time.Sleep(60 * time.Second)
}
func main() {
	fmt.Println("main start")
	defer test1()
	test2()
	fmt.Println("main end")
}

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

main start

Process finished with the exit code -1073741510 (0xC000013A: interrupted by Ctrl+C)

如上,我們test2() 睡眠時(shí)間內(nèi),點(diǎn)擊Ctrl+C,發(fā)現(xiàn)defer test1()并沒有執(zhí)行。

總結(jié):這個(gè)點(diǎn)很重要,需要我們?cè)谌粘.惓V袛鄷r(shí),留意defer是否未處理的情況。
所以一般情況下,我們程序需要捕獲這種異常中斷,在程序退出前,手動(dòng)做一些處理。

參考文獻(xiàn)

Go常見坑:Go語(yǔ)言里被defer的函數(shù)一定會(huì)執(zhí)行么?
參考URL: https://blog.csdn.net/perfumekristy/article/details/121343642
面試官:聽說你精通golang的defer?
參考URL: https://cloud.tencent.com/developer/article/2076951

到此這篇關(guān)于golang中的defer函數(shù)理解的文章就介紹到這了,更多相關(guān)golang defer函數(shù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • go語(yǔ)言實(shí)現(xiàn)全排列的示例代碼

    go語(yǔ)言實(shí)現(xiàn)全排列的示例代碼

    本文主要介紹了go語(yǔ)言實(shí)現(xiàn)全排列的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-03-03
  • Go中strings包的基本使用示例代碼

    Go中strings包的基本使用示例代碼

    本文詳細(xì)介紹了Go語(yǔ)言中strings包的基本使用方法,包括字符串的前綴、后綴判斷,字符串包含、索引查找、字符串替換、計(jì)數(shù)、重復(fù)、大小寫轉(zhuǎn)換、修剪、分割、拼接以及數(shù)據(jù)類型轉(zhuǎn)換等功能,示例代碼豐富,適合初學(xué)者和需要使用字符串處理功能的開發(fā)者參考學(xué)習(xí)
    2024-10-10
  • 詳解Golang中Context的三個(gè)常見應(yīng)用場(chǎng)景

    詳解Golang中Context的三個(gè)常見應(yīng)用場(chǎng)景

    Golang?context主要用于定義超時(shí)取消,取消后續(xù)操作,在不同操作中傳遞值。本文通過簡(jiǎn)單易懂的示例進(jìn)行說明,感興趣的可以了解一下
    2022-12-12
  • Go語(yǔ)言循環(huán)遍歷含有中文的字符串的方法小結(jié)

    Go語(yǔ)言循環(huán)遍歷含有中文的字符串的方法小結(jié)

    這篇文章主要介紹了Go語(yǔ)言循環(huán)遍歷含有中文的字符串的幾種方法,文章通過代碼示例講解的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴跟著小編一起來看看吧
    2023-07-07
  • go語(yǔ)言中切片與內(nèi)存復(fù)制 memcpy 的實(shí)現(xiàn)操作

    go語(yǔ)言中切片與內(nèi)存復(fù)制 memcpy 的實(shí)現(xiàn)操作

    這篇文章主要介紹了go語(yǔ)言中切片與內(nèi)存復(fù)制 memcpy 的實(shí)現(xiàn)操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2021-04-04
  • 使用Go重試機(jī)制代碼更可靠

    使用Go重試機(jī)制代碼更可靠

    這篇文章主要為大家介紹了使用Go重試機(jī)制的使用,使你的代碼更加可靠,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-08-08
  • 夯實(shí)Golang基礎(chǔ)之?dāng)?shù)據(jù)類型梳理匯總

    夯實(shí)Golang基礎(chǔ)之?dāng)?shù)據(jù)類型梳理匯總

    這篇文章主要8為大家介紹了夯實(shí)Golang基礎(chǔ)之?dāng)?shù)據(jù)類型梳理匯總,有需要的朋友可以借鑒參考下,希望能夠有所幫助
    2023-10-10
  • Go語(yǔ)言學(xué)習(xí)筆記之文件讀寫操作詳解

    Go語(yǔ)言學(xué)習(xí)筆記之文件讀寫操作詳解

    這篇文章主要為大家詳細(xì)介紹了Go語(yǔ)言對(duì)文件進(jìn)行讀寫操作的方法,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)Go語(yǔ)言有一定的幫助,需要的可以參考一下
    2022-05-05
  • Go語(yǔ)言流程控制詳情

    Go語(yǔ)言流程控制詳情

    這篇文章主要介紹了Go語(yǔ)言流程控制詳情,流程控制包含分三大類:條件判斷,循環(huán)控制和無條件跳轉(zhuǎn)。下面關(guān)于更多相關(guān)內(nèi)容需要的小伙伴可以參考一下
    2022-03-03
  • 一文帶你了解Go語(yǔ)言標(biāo)準(zhǔn)庫(kù)strings的常用函數(shù)和方法

    一文帶你了解Go語(yǔ)言標(biāo)準(zhǔn)庫(kù)strings的常用函數(shù)和方法

    strings?庫(kù)包含了許多高效的字符串常用操作的函數(shù)和方法,巧用這些函數(shù)與方法,能極大的提高我們程序的性能。本文就來和大家分享一下Go標(biāo)準(zhǔn)庫(kù)strings的常用函數(shù)和方法,希望對(duì)大家有所幫助
    2022-11-11

最新評(píng)論