使用Go語(yǔ)言中的Context取消協(xié)程執(zhí)行的操作代碼
使用 Go 語(yǔ)言中的 Context 取消協(xié)程執(zhí)行
在 Go 語(yǔ)言中,協(xié)程(goroutine)是一種輕量級(jí)的線程,非常適合處理并發(fā)任務(wù)。然而,如何優(yōu)雅地取消正在運(yùn)行的協(xié)程是一個(gè)常見(jiàn)的問(wèn)題。本文將通過(guò)一個(gè)具體的例子來(lái)展示如何使用 context
包來(lái)取消協(xié)程的執(zhí)行,特別是處理嵌套任務(wù)中的取消問(wèn)題。
問(wèn)題描述
假設(shè)我們有一個(gè)長(zhǎng)時(shí)間運(yùn)行的任務(wù),該任務(wù)包含一個(gè)外層循環(huán)和一個(gè)內(nèi)層任務(wù)。我們需要在外層循環(huán)接收到取消信號(hào)時(shí),能夠立即終止內(nèi)層任務(wù)。以下是一個(gè)示例代碼:
package main import ( "context" "fmt" "time" ) // longRunningTask 是一個(gè)模擬長(zhǎng)時(shí)間運(yùn)行的任務(wù)。 func longRunningTask(ctx context.Context) { for { select { case <-ctx.Done(): // 監(jiān)聽(tīng) ctx.Done() 以獲取取消信號(hào) fmt.Println("任務(wù)被取消:", ctx.Err()) return // 接收到取消信號(hào)后退出 default: currentTime := time.Now().Format("2006-01-02 15:04:05") // 獲取并格式化當(dāng)前時(shí)間 fmt.Printf("任務(wù)進(jìn)行中... 當(dāng)前時(shí)間:%s\n", currentTime) for { fmt.Printf("111") time.Sleep(1 * time.Second) // } } } } func main() { // 創(chuàng)建一個(gè)可以取消的 context ctx, cancel := context.WithCancel(context.Background()) // 啟動(dòng)一個(gè)新的 goroutine 執(zhí)行任務(wù) go longRunningTask(ctx) // 模擬一段時(shí)間后取消任務(wù) time.Sleep(3 * time.Second) fmt.Println("取消任務(wù)...") cancel() // 發(fā)送取消信號(hào) // 等待一段時(shí)間讓任務(wù)有時(shí)間處理取消信號(hào)并退出 time.Sleep(10 * time.Second) }
在這個(gè)示例中,當(dāng)我們?nèi)∠蝿?wù)時(shí),外層循環(huán)會(huì)接收到取消信號(hào)并退出,但內(nèi)層循環(huán)會(huì)繼續(xù)運(yùn)行,因?yàn)槲覀儧](méi)有在內(nèi)層循環(huán)中檢查取消信號(hào)。
解決方案
為了確保內(nèi)層任務(wù)也能響應(yīng)取消信號(hào),我們需要在內(nèi)層任務(wù)中也檢查 ctx.Done() 通道。以下是修改后的代碼:
package main import ( "context" "fmt" "time" ) // longRunningTask 是一個(gè)模擬長(zhǎng)時(shí)間運(yùn)行的任務(wù)。 func longRunningTask(ctx context.Context) { for { select { case <-ctx.Done(): // 監(jiān)聽(tīng) ctx.Done() 以獲取取消信號(hào) fmt.Println("任務(wù)被取消:", ctx.Err()) return // 接收到取消信號(hào)后退出 default: currentTime := time.Now().Format("2006-01-02 15:04:05") // 獲取并格式化當(dāng)前時(shí)間 fmt.Printf("任務(wù)進(jìn)行中... 當(dāng)前時(shí)間:%s\n", currentTime) // 啟動(dòng)內(nèi)層任務(wù) runInnerTask(ctx) } } } // runInnerTask 是一個(gè)模擬內(nèi)層長(zhǎng)時(shí)間運(yùn)行的任務(wù)。 func runInnerTask(ctx context.Context) { for { select { case <-ctx.Done(): // 內(nèi)層任務(wù)也監(jiān)聽(tīng) ctx.Done() fmt.Println("內(nèi)層任務(wù)被取消:", ctx.Err()) return // 接收到取消信號(hào)后退出 default: fmt.Printf("111") time.Sleep(1 * time.Second) } } } func main() { // 創(chuàng)建一個(gè)可以取消的 context ctx, cancel := context.WithCancel(context.Background()) // 啟動(dòng)一個(gè)新的 goroutine 執(zhí)行任務(wù) go longRunningTask(ctx) // 模擬一段時(shí)間后取消任務(wù) time.Sleep(3 * time.Second) fmt.Println("取消任務(wù)...") cancel() // 發(fā)送取消信號(hào) // 等待一段時(shí)間讓任務(wù)有時(shí)間處理取消信號(hào)并退出 time.Sleep(10 * time.Second) }
解釋
外層循環(huán):
外層循環(huán)使用 select 語(yǔ)句來(lái)監(jiān)聽(tīng) ctx.Done() 通道。如果接收到取消信號(hào),任務(wù)會(huì)打印一條消息并退出。
內(nèi)層任務(wù):
內(nèi)層任務(wù)也使用 select 語(yǔ)句來(lái)監(jiān)聽(tīng) ctx.Done() 通道。如果接收到取消信號(hào),內(nèi)層任務(wù)會(huì)打印一條消息并退出。
通過(guò)這種方式,我們可以確保無(wú)論是在外層循環(huán)還是內(nèi)層任務(wù)中,任務(wù)都能響應(yīng)取消信號(hào)并優(yōu)雅地退出。
總結(jié)
在 Go 語(yǔ)言中,使用 context 包來(lái)管理協(xié)程的生命周期是非常重要的。通過(guò)在每個(gè)需要響應(yīng)取消信號(hào)的地方檢查 ctx.Done() 通道,我們可以確保任務(wù)能夠及時(shí)響應(yīng)取消信號(hào)并優(yōu)雅地退出。這對(duì)于構(gòu)建健壯和可靠的并發(fā)應(yīng)用程序至關(guān)重要。
到此這篇關(guān)于使用Go語(yǔ)言中的Context取消協(xié)程執(zhí)行的操作代碼的文章就介紹到這了,更多相關(guān)Go Context取消協(xié)程執(zhí)行內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
IdeaGo啟動(dòng)報(bào)錯(cuò)Failed to create JVM的問(wèn)題解析
這篇文章主要介紹了IdeaGo啟動(dòng)報(bào)錯(cuò)Failed to create JVM的問(wèn)題,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-11-11GO語(yǔ)言協(xié)程互斥鎖Mutex和讀寫(xiě)鎖RWMutex用法實(shí)例詳解
這篇文章主要介紹了GO語(yǔ)言協(xié)程互斥鎖Mutex和讀寫(xiě)鎖RWMutex用法詳解,需要的朋友可以參考下2022-04-04向Rust學(xué)習(xí)Go考慮簡(jiǎn)單字符串插值特性示例解析
這篇文章主要為大家介紹了向Rust學(xué)習(xí)Go考慮簡(jiǎn)單字符串插值特性示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-02-02Go實(shí)現(xiàn)跨平臺(tái)的藍(lán)牙聊天室示例詳解
這篇文章主要為大家介紹了Go實(shí)現(xiàn)跨平臺(tái)的藍(lán)牙聊天室示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-12-12一文帶你熟悉Go語(yǔ)言中的分支結(jié)構(gòu)
這篇文章主要和大家分享一下Go語(yǔ)言中的分支結(jié)構(gòu)(if?-?else-if?-?else、switch),文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)Go語(yǔ)言有一定的幫助,需要的可以參考一下2022-11-11Golang在整潔架構(gòu)基礎(chǔ)上實(shí)現(xiàn)事務(wù)操作
這篇文章在 go-kratos 官方的 layout 項(xiàng)目的整潔架構(gòu)基礎(chǔ)上,實(shí)現(xiàn)優(yōu)雅的數(shù)據(jù)庫(kù)事務(wù)操作,需要的朋友可以參考下2024-08-08Golang官方限流器time/rate的使用與實(shí)現(xiàn)詳解
限流器是后臺(tái)服務(wù)中十分重要的組件,在實(shí)際的業(yè)務(wù)場(chǎng)景中使用居多。time/rate?包基于令牌桶算法實(shí)現(xiàn)限流,本文主要為大家介紹了time/rate的使用與實(shí)現(xiàn),需要的可以參考一下2023-04-04golang中sync.Mutex的實(shí)現(xiàn)方法
本文主要介紹了golang中sync.Mutex的實(shí)現(xiàn)方法,mutex?主要有兩個(gè)?method:?Lock()?和?Unlock(),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-04-04