Go defer與time.sleep的使用與區(qū)別
請(qǐng)大家看下面這段代碼,看運(yùn)行結(jié)果會(huì)出現(xiàn)什么,為什么?
問(wèn)題
demo
package main
import (
"log"
"time"
)
func main() {
start := time.Now()
defer func() {
log.Printf("匿名函數(shù)時(shí)間差: %v", time.Since(start))
}()
defer log.Printf("時(shí)間差: %v", time.Since(start))
time.Sleep(3 * time.Second)
log.Printf("函數(shù)結(jié)束")
}
這里不少同學(xué)會(huì)認(rèn)為:這題我會(huì),先輸出
函數(shù)結(jié)束,待整個(gè)函數(shù)結(jié)束后,根據(jù)defer后進(jìn)先出的順序依次打印計(jì)算的時(shí)間差,由于這里睡眠了3秒,兩個(gè)時(shí)間差應(yīng)該都是3秒左右。
所以答案為:
函數(shù)結(jié)束
時(shí)間差:3s
匿名函數(shù)時(shí)間差:3s
看到這里,我想說(shuō)同學(xué)你的思路和想法是好的,一開(kāi)始我也和你一樣,但是這里的答案是錯(cuò)的,那為什么錯(cuò)呢?總得有個(gè)原因吧!很明顯匿名函數(shù)的時(shí)間差符合邏輯是對(duì)的,那為什么時(shí)間差與預(yù)期不符合呢?下面我來(lái)進(jìn)行分析。
運(yùn)行結(jié)果

分析
這里的關(guān)鍵點(diǎn)在于為什么時(shí)間差為0,也就是說(shuō)為什么時(shí)間差這個(gè)defer語(yǔ)句沒(méi)有被time.sleep 所影響?進(jìn)一步分析,就是查看start的賦值時(shí)機(jī)在哪?是在一開(kāi)始調(diào)用defer就賦值,還是說(shuō)在函數(shù)結(jié)束后給defer中的start賦值,從而造成結(jié)果的不同。
帶著這個(gè)問(wèn)題,不妨來(lái)debug一下,看一下函數(shù)語(yǔ)句的執(zhí)行順序。
設(shè)置斷點(diǎn)
要想看一下start的賦值時(shí)機(jī),設(shè)置斷點(diǎn)在出現(xiàn)start變量前面即可。

設(shè)置好斷點(diǎn)后,開(kāi)始進(jìn)行debug
debug查看
step1
一開(kāi)始,先初始化start的值

step2
接著,來(lái)到第一個(gè)defer + 匿名函數(shù),看它有沒(méi)有進(jìn)去里面的printf語(yǔ)句給start變量進(jìn)行賦值,step over 直接跳過(guò)了匿名函數(shù)的defer語(yǔ)句,也就是說(shuō)明并沒(méi)有給匿名函數(shù)中的defer語(yǔ)句中的start賦值

step3
之后,進(jìn)入defer語(yǔ)句中,給time.Since中的start進(jìn)行賦值。
這里,會(huì)發(fā)現(xiàn)問(wèn)題的關(guān)鍵所在,對(duì)比defer + 匿名函數(shù) 一開(kāi)始調(diào)用(非執(zhí)行)時(shí),不會(huì)對(duì)里面的變量(參數(shù))start進(jìn)行賦值。
然而普通的defer + printf則在一開(kāi)始時(shí)就會(huì)對(duì)里面的變量start賦值,賦值后不會(huì)先把time.since的結(jié)果計(jì)算出來(lái),會(huì)在defer調(diào)用后,也就是光標(biāo)移動(dòng)到下一條語(yǔ)句時(shí)調(diào)用time.sleep計(jì)算時(shí)間差,其結(jié)果就是0s 左右,此時(shí)不會(huì)在調(diào)用時(shí)輸出,待整個(gè)函數(shù)執(zhí)行完畢退出后,依次按照defer 順序輸出。

step4
進(jìn)一步來(lái)到了time.sleep ,這也說(shuō)明step3 確實(shí)給defer 語(yǔ)句賦值了,并沒(méi)有跳過(guò),現(xiàn)在開(kāi)始休眠3 s, 觀察3s后會(huì)光標(biāo)會(huì)去到哪里?
猜想:3s 后會(huì)先輸出函數(shù)結(jié)束 ,之后函數(shù)退出,開(kāi)始執(zhí)行defer 語(yǔ)句.
結(jié)合匿名函數(shù)的時(shí)間差為3s左右,又因?yàn)?code>defer +匿名函數(shù)中的start 還未賦值,會(huì)回到開(kāi)頭的defer + 匿名函數(shù)進(jìn)行賦值。

等待3s鐘

step5
3s后,來(lái)到了輸出函數(shù)結(jié)束的語(yǔ)句。

進(jìn)一步函數(shù)退出,也就是整個(gè)函數(shù)執(zhí)行完畢!

step6
又回到了剛才的defer + 匿名函數(shù)果然與step4的猜想一致!

關(guān)鍵點(diǎn)來(lái)了!如下圖:這里會(huì)進(jìn)入defer + 匿名函數(shù) , 并給里面的start變量賦值。
注意,這里傳參start 還是一開(kāi)始的start! 只不過(guò)time.since(start) 的time 增加(休眠)了3s ,這也很好的解釋了為什么defer + 匿名函數(shù) 輸出的是3s 左右,而defer + printf 語(yǔ)句卻是輸出0s 左右。

之后匿名函數(shù)結(jié)束

退出當(dāng)前的整個(gè)函數(shù)

最后整個(gè)函數(shù)結(jié)束,輸出debug 的結(jié)果:這里加了一些debug 調(diào)試的時(shí)間,以運(yùn)行結(jié)果為準(zhǔn),見(jiàn)下。

運(yùn)行結(jié)果如下:

探討
在debug 后,我們來(lái)探討一下為什么會(huì)出現(xiàn)這樣的情況?為什么結(jié)果會(huì)有所不同?里面的機(jī)制是什么?
defer + 輸出語(yǔ)句
傳值時(shí)機(jī):一開(kāi)始調(diào)用defer 時(shí)傳入
Go 語(yǔ)言中所有的函數(shù)調(diào)用都是傳值的
雖然 defer 是關(guān)鍵字,但是也繼承了這個(gè)特性。假設(shè)我們想要計(jì)算 main 函數(shù)運(yùn)行的時(shí)間,可能會(huì)寫(xiě)出以下的代碼:
package main
import (
"fmt"
"time"
)
func main() {
start := time.Now()
// 這里誤以為:startedAt是在time.Sleep之后才會(huì)將參數(shù)傳遞給defer所在語(yǔ)句的函數(shù)中
defer fmt.Println(time.Since(start))
time.Sleep(3 * time.Second)
}
關(guān)鍵點(diǎn):調(diào)用defer關(guān)鍵字會(huì)立刻拷貝函數(shù)中引用的外部參數(shù)
所以 time.Since(start) 的結(jié)果不是在 main 函數(shù)退出之前計(jì)算的,而是在 defer 關(guān)鍵字調(diào)用時(shí)賦值計(jì)算的,最終導(dǎo)致上述代碼輸出 0s。
defer + 匿名函數(shù)
傳值時(shí)機(jī):main函數(shù)結(jié)束后,執(zhí)行defer函數(shù) 時(shí)傳入,傳入函數(shù)指針
package main
import (
"fmt"
"time"
)
func main() {
start := time.Now()
// 使用匿名函數(shù),傳遞的是函數(shù)的指針
defer func() {
fmt.Println(time.Since(start))
}()
time.Sleep(3 * time.Second)
}
那為什么使用匿名函數(shù)就可以輸出3s 呢?關(guān)鍵點(diǎn):defer 使用匿名函數(shù) , 傳遞的是函數(shù)的指針(函數(shù)是一種指針類(lèi)型),結(jié)合剛才的斷點(diǎn)分析,不會(huì)在一開(kāi)始調(diào)用defer + 匿名函數(shù)的時(shí)候就直接賦值,而是在后面main函數(shù)退出時(shí),再執(zhí)行defer +匿名函數(shù) 時(shí)再傳入函數(shù)的指針,并給start變量賦值。
總結(jié)
總而言之:一開(kāi)始調(diào)用defer+輸出語(yǔ)句會(huì)進(jìn)行傳值,會(huì)在調(diào)用defer 的時(shí)候就直接傳遞值的拷貝(如剛才的defer+輸出語(yǔ)句),從而計(jì)算時(shí)間差。
調(diào)用defer + 匿名函數(shù) 傳遞的是指針類(lèi)型(如函數(shù)、匿名函數(shù)), 會(huì)在main函數(shù)退出時(shí),在執(zhí)行defer +匿名函數(shù) 時(shí)再傳入函數(shù)的指針,并給里面的變量賦值。
歸根結(jié)底,是傳入的類(lèi)型不同,繼而導(dǎo)致傳入變量的時(shí)機(jī)不同,造成輸出的結(jié)果不同!
擴(kuò)展
面試官:請(qǐng)你用defer與time.sleep寫(xiě)一個(gè)計(jì)算函數(shù)運(yùn)行時(shí)間的程序
相信看到這里的小伙伴,已經(jīng)很清楚要寫(xiě)什么代碼了!
package main
import (
"fmt"
"time"
)
func main() {
start := time.Now()
// 使用匿名函數(shù),傳遞的是函數(shù)的指針
defer func() {
fmt.Println(time.Since(start))
}()
time.Sleep(3 * time.Second)
}到此這篇關(guān)于Go defer與time.sleep的使用與區(qū)別的文章就介紹到這了,更多相關(guān)Go defer與time.sleep內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Golang中HTTP路由設(shè)計(jì)的使用與實(shí)現(xiàn)
這篇文章主要介紹了Golang中HTTP路由設(shè)計(jì)的使用與實(shí)現(xiàn),為什么要設(shè)計(jì)路由規(guī)則,因?yàn)槁酚梢?guī)則是HTTP的請(qǐng)求按照一定的規(guī)則 ,匹配查找到對(duì)應(yīng)的控制器并傳遞執(zhí)行的邏輯,需要的朋友可以參考下2023-05-05
gin項(xiàng)目部署到服務(wù)器并后臺(tái)啟動(dòng)的步驟
本文主要介紹了gin項(xiàng)目部署到服務(wù)器并后臺(tái)啟動(dòng)的步驟,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-02-02
淺析go語(yǔ)言如何實(shí)現(xiàn)協(xié)程的搶占式調(diào)度的
go語(yǔ)言通過(guò)GMP模型實(shí)現(xiàn)協(xié)程并發(fā),為了避免單協(xié)程持續(xù)持有線(xiàn)程導(dǎo)致線(xiàn)程隊(duì)列中的其他協(xié)程饑餓問(wèn)題,設(shè)計(jì)者提出了一個(gè)搶占式調(diào)度機(jī)制,本文會(huì)基于一個(gè)簡(jiǎn)單的代碼示例對(duì)搶占式調(diào)度過(guò)程進(jìn)行深入講解剖析2024-04-04
golang?pprof監(jiān)控memory?block?mutex統(tǒng)計(jì)原理分析
這篇文章主要為大家介紹了golang?pprof監(jiān)控memory?block?mutex統(tǒng)計(jì)原理分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-04-04

