golang中context使用小結(jié)
一、context使用注意事項(xiàng)
在使用context時(shí),有一些需要注意的事項(xiàng),以及一些與性能優(yōu)化相關(guān)的建議:
避免濫用context傳遞數(shù)據(jù):context的主要目的是傳遞請(qǐng)求范圍的數(shù)據(jù)和取消信號(hào),而不是用于傳遞全局狀態(tài)或大量數(shù)據(jù)。濫用context傳遞大量數(shù)據(jù)可能導(dǎo)致上下文對(duì)象變得臃腫,增加內(nèi)存和GC壓力。
不要修改已傳遞的context:傳遞的context是不可變的,即使在函數(shù)內(nèi)部對(duì)其調(diào)用cancel方法也不會(huì)影響調(diào)用方的context。如果需要對(duì)context進(jìn)行修改,應(yīng)該通過(guò)返回一個(gè)新的派生context來(lái)實(shí)現(xiàn)。
只在需要時(shí)傳遞context:不要將context作為函數(shù)參數(shù)無(wú)限制地傳遞,而是在需要時(shí)傳遞。這樣可以避免不必要的復(fù)雜性和代碼膨脹。
及早檢查取消信號(hào):在使用context的地方,應(yīng)該及早檢查
ctx.Done()
的返回值,以盡早響應(yīng)取消信號(hào)。在耗時(shí)操作前或可能阻塞的地方,應(yīng)該通過(guò)select語(yǔ)句來(lái)監(jiān)聽(tīng)多個(gè)操作,包括取消信號(hào)、超時(shí)和其他channel。使用WithCancel替代WithTimeout:在可能的情況下,優(yōu)先使用WithCancel函數(shù)來(lái)設(shè)置取消信號(hào),而不是僅僅依賴于WithTimeout函數(shù)。這樣可以有更精確的控制和更靈活的處理方式。
優(yōu)化context的傳遞:在頻繁調(diào)用的函數(shù)鏈中,避免在每個(gè)函數(shù)中重復(fù)傳遞相同的context,可以通過(guò)使用結(jié)構(gòu)體或函數(shù)閉包將context作為參數(shù)進(jìn)行傳遞,從而減少代碼重復(fù)和提升性能。
及時(shí)取消不再需要的goroutine:如果在多個(gè)goroutine中使用context,確保在不再需要時(shí)及時(shí)取消goroutine,以避免資源浪費(fèi)和潛在的goroutine泄漏。
這些注意事項(xiàng)和性能優(yōu)化建議可幫助確保正確且高效地使用context,避免濫用和性能問(wèn)題。根據(jù)具體場(chǎng)景和需求,可以靈活使用context的機(jī)制來(lái)優(yōu)化代碼的可讀性、并發(fā)安全性和性能。
二、context使用舉例
在Go語(yǔ)言中,context(上下文)是在不同goroutine之間傳遞請(qǐng)求范圍數(shù)據(jù)、取消信號(hào)和超時(shí)處理的一種機(jī)制。下面詳細(xì)介紹context的每種使用情況和相應(yīng)的代碼舉例:
傳遞請(qǐng)求范圍數(shù)據(jù):
package main import ( "context" "fmt" ) // 定義一個(gè)鍵類型(key)用于context中的數(shù)據(jù)傳遞 type key string // 在context中設(shè)置數(shù)據(jù) func withValue(ctx context.Context) { // 使用WithValue將數(shù)據(jù)存儲(chǔ)在context中 ctxWithData := context.WithValue(ctx, key("name"), "John") // 調(diào)用另一個(gè)函數(shù),并將帶有數(shù)據(jù)的context傳遞給它 printName(ctxWithData) } // 從context中獲取并使用數(shù)據(jù) func printName(ctx context.Context) { // 從context中獲取數(shù)據(jù),并進(jìn)行類型斷言 if name, ok := ctx.Value(key("name")).(string); ok { fmt.Println("Name:", name) } } func main() { // 創(chuàng)建根context ctx := context.Background() // 傳遞context并設(shè)置數(shù)據(jù) withValue(ctx) }
在上面的示例中,我們定義了一個(gè)
key
類型,用于在context中存儲(chǔ)數(shù)據(jù)。然后,我們使用WithValue
函數(shù)將數(shù)據(jù)存儲(chǔ)在帶有數(shù)據(jù)的contextctxWithData
中,并將其傳遞給printName
函數(shù)。在printName
函數(shù)中,我們使用Value
方法從context中獲取數(shù)據(jù),并進(jìn)行類型斷言后打印出來(lái)。取消信號(hào):
package main import ( "context" "fmt" "time" ) // 模擬一些耗時(shí)操作 func performTask(ctx context.Context) { // 檢查是否接收到取消信號(hào) select { case <-ctx.Done(): fmt.Println("Task canceled") return default: // 模擬長(zhǎng)時(shí)間運(yùn)行的任務(wù) time.Sleep(5 * time.Second) fmt.Println("Task completed") } } func main() { // 創(chuàng)建根context ctx := context.Background() // 派生子context,并設(shè)置取消信號(hào) ctx, cancel := context.WithCancel(ctx) // 啟動(dòng)耗時(shí)操作的goroutine,并傳遞帶有取消信號(hào)的context go performTask(ctx) // 模擬一些操作后取消任務(wù) time.Sleep(2 * time.Second) cancel() // 發(fā)送取消信號(hào) // 等待一段時(shí)間,確保程序有足夠的時(shí)間處理取消信號(hào) time.Sleep(1 * time.Second) }
在上面的示例中,我們創(chuàng)建了一個(gè)任務(wù)函數(shù)
performTask
,該函數(shù)會(huì)檢查是否接收到取消信號(hào)。使用context.WithCancel
函數(shù)創(chuàng)建派生的子context,并通過(guò)調(diào)用返回的cancel
函數(shù)發(fā)送取消信號(hào)。然后,我們?cè)谝粋€(gè)goroutine中運(yùn)行任務(wù)函數(shù),并通過(guò)傳遞帶有取消信號(hào)的context來(lái)監(jiān)聽(tīng)取消信號(hào)。在主goroutine中,我們等待一段時(shí)間后調(diào)用cancel
函數(shù)發(fā)送取消信號(hào)。當(dāng)任務(wù)函數(shù)接收到取消信號(hào)后,它會(huì)打印"Task canceled"。超時(shí)處理:
package main import ( "context" "fmt" "time" ) // 模擬一些耗時(shí)操作 func performTask(ctx context.Context) { // 檢查是否接收到取消信號(hào)或超時(shí) select { case <-ctx.Done(): fmt.Println("Task canceled") case <-time.After(5 * time.Second): fmt.Println("Task completed") } } func main() { // 創(chuàng)建根context ctx := context.Background() // 派生子context,并設(shè)置超時(shí)時(shí)間 ctx, cancel := context.WithTimeout(ctx, 3*time.Second) defer cancel() // 啟動(dòng)耗時(shí)操作的goroutine,并傳遞帶有超時(shí)設(shè)置的context go performTask(ctx) // 等待一段時(shí)間,確保程序有足夠的時(shí)間處理超時(shí)或取消信號(hào) time.Sleep(5 * time.Second) }
在上面的示例中,我們創(chuàng)建了一個(gè)任務(wù)函數(shù)
performTask
,該函數(shù)會(huì)檢查是否接收到取消信號(hào)或超時(shí)。使用context.WithTimeout
函數(shù)創(chuàng)建派生的子context,并通過(guò)調(diào)用返回的cancel
函數(shù)來(lái)設(shè)置超時(shí)時(shí)間。然后,我們?cè)谝粋€(gè)goroutine中運(yùn)行任務(wù)函數(shù),并傳遞帶有超時(shí)設(shè)置的context來(lái)監(jiān)聽(tīng)超時(shí)或取消信號(hào)。在主goroutine中,我們等待一段時(shí)間以確保程序有足夠的時(shí)間處理超時(shí)或取消信號(hào)。當(dāng)超過(guò)超時(shí)時(shí)間后,任務(wù)函數(shù)會(huì)打印"Task canceled"。
這些是context在Go語(yǔ)言中的常見(jiàn)用法,它們使得在并發(fā)環(huán)境中處理請(qǐng)求范圍數(shù)據(jù)、取消信號(hào)和超時(shí)變得更加簡(jiǎn)單和可靠。根據(jù)具體的使用場(chǎng)景,你可以選擇適當(dāng)?shù)腸ontext函數(shù)來(lái)創(chuàng)建和傳遞context,并根據(jù)需要進(jìn)行取消和超時(shí)處理。
到此這篇關(guān)于golang中context使用小結(jié)的文章就介紹到這了,更多相關(guān)golang context 內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
GO語(yǔ)言中Chan實(shí)現(xiàn)原理的示例詳解
這篇文章主要為大家詳細(xì)介紹了Go語(yǔ)言中Chan實(shí)現(xiàn)原理的相關(guān)資料,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)Go語(yǔ)言有一定的幫助,需要的可以參考一下2023-02-02一文帶你了解Go語(yǔ)言實(shí)現(xiàn)的并發(fā)神庫(kù)conc
前幾天逛github發(fā)現(xiàn)了一個(gè)有趣的并發(fā)庫(kù)-conc,這篇文章將為大家詳細(xì)介紹一下這個(gè)庫(kù)的實(shí)現(xiàn),文中的示例代碼講解詳細(xì),感興趣的可以了解一下2023-01-01Go語(yǔ)言學(xué)習(xí)之條件語(yǔ)句使用詳解
這篇文章主要介紹了Go語(yǔ)言中條件語(yǔ)句的使用,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-04-04Go語(yǔ)言擴(kuò)展原語(yǔ)之Semaphore的用法詳解
Go語(yǔ)言的擴(kuò)展包中提供了帶權(quán)重的信號(hào)量?semaphore.Weighted,讓我們可以按照不同的權(quán)重管理資源的訪問(wèn),下面小編就來(lái)和大家聊聊它們的具體用法吧2023-07-07