自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Golang中Context包使用場景和示例詳解

開發(fā) 前端
Context包提供了一種機制,可以在多個Goroutine之間進行通信和控制。使用Context包能夠有效地控制程序的并發(fā)性,提高程序的健壯性和性能。

本文結(jié)合示例代碼講解一下context包的幾種使用場景。

控制子協(xié)程退出

context包提供了一種機制,可以在多個goroutine之間進行通信和控制。使用Context包能夠有效地控制程序的并發(fā)性,提高程序的健壯性和性能。

Golang是沒有辦法讓其他goroutine退出的,goroutine只能自己退出。之所以說context包可以控制子協(xié)程退出意思是子協(xié)程可以接收到主協(xié)程發(fā)出的退出信號,然后自己退出??慈缦率纠a:

package main

import (
	"context"
	"errors"
	"sync"
)

func request(ctx context.Context, url string) error {
	result := make(chan int)
	err := make(chan error)
	go func() {
		// 假如isSuccess是請求返回的結(jié)果,成功則通過result傳遞成功信息,錯誤通過error傳遞錯誤信息
		isSuccess := true
		if isSuccess {
			result <- 1
		} else {
			err <- errors.New("some error happen")
		}
	}()

	select {
	case <-ctx.Done():
		// 其他請求失敗
		return ctx.Err()
	case e := <-err:
		// 本次請求失敗,返回錯誤信息
		return e
	case <-result:
		// 本此請求成功,不返回錯誤信息
		return nil
	}
}

func main() {
	ctx, cancel := context.WithCancel(context.Background())
	// 調(diào)用接口a
	err := request(ctx, "https://xxx.com/a")
	if err != nil {
		return
	}
	wg := sync.WaitGroup{}
	// 調(diào)用接口b
	wg.Add(1)
	go func() {
		defer wg.Done()
		err := request(ctx, "https://xxx.com/b")
		if err != nil {
			cancel()
		}
	}()
	// 調(diào)用接口c
	wg.Add(1)
	go func() {
		defer wg.Done()
		err := request(ctx, "https://xxx.com/c")
		if err != nil {
			cancel()
		}
	}()
	wg.Wait()
}

首先調(diào)用context.WithCancel方法構(gòu)造了一個Context和返回了一個cancel函數(shù),其他goroutine調(diào)用的方法都傳入了這個Context作為第一個參數(shù),當(dāng)主goroutine想要告訴所有g(shù)oroutine需要退出的時候,通過調(diào)用cancel函數(shù)把退出的信息告訴所有的goroutine。所有g(shù)oroutine通過監(jiān)聽ctx.Done返回的channel得到退出信號然后退出。

超時控制

例如查詢數(shù)據(jù)庫、調(diào)用RPC服務(wù)、調(diào)用HTTP接口等場景,這些操作都是阻塞的,如果一直不返回數(shù)據(jù)的話,會影響產(chǎn)品的用戶體驗。針對這些情況的解決方式通常是設(shè)置一個超時間,超過后自動取消操作。

使用context包中的WithDeadline和WithTimeout方法可以實現(xiàn)這個解決方法。先看如下例子:

package main

import (
    "context"
    "fmt"
    "time"
)

func main() {
    ctx, cancel := context.WithTimeout(context.Background(), 50*time.Millisecond)
    defer cancel()

    select {
    case <-time.After(1 * time.Second):
        fmt.Println("overslept")
    case <-ctx.Done():
        fmt.Println(ctx.Err()) // 輸出 "context deadline exceeded"
    }
}

因為設(shè)置的超時時間是50毫秒,所以select會進入第二個case,會輸出“context deadline exceeded”。

Golang的net/http包發(fā)起http請求的時候是實現(xiàn)了超時控制的,看如下代碼:

package main

import (
	"context"
	"io"
	"log"
	"net/http"
	"time"
)

func main() {
	req, err := http.NewRequest(http.MethodGet, "https://www.baidu.com", nil)
	if err != nil {
		log.Fatal(err)
	}

	// 構(gòu)造一個超時間為50毫秒的Context
	ctx, cancel := context.WithTimeout(context.Background(), 50*time.Millisecond)
	defer cancel()
	req = req.WithContext(ctx)
	
	c := &http.Client{}
	res, err := c.Do(req)
	if err != nil {
		log.Fatal(err)
	}
	defer res.Body.Close()
	out, err := io.ReadAll(res.Body)
	if err != nil {
		log.Fatal(err)
	}
	log.Println(string(out))
}

執(zhí)行后會輸出“context deadline exceeded”,如果將context.WithTimeout方法的timeout參數(shù)調(diào)大一些,就可以看到正常的返回數(shù)據(jù)。

上下文傳遞數(shù)據(jù)

這個作用在鏈路追蹤中非常重要,鏈路追蹤需要將traceID層層往下傳遞,在服務(wù)間傳遞。

type traceIdKey struct{}{}

// 定義固定的Key
var TraceIdKey = traceIdKey{}

func ServeHTTP(w http.ResponseWriter, req *http.Request){
  // 首先從請求中獲取到traceID
  traceId := getTraceIdFromRequest(req)
  // 將Key存入Context中
  ctx := context.WithValue(req.Context(), TraceIdKey, traceId)
  // 設(shè)置超時時間
  ctx = context.WithTimeout(ctx, time.Second)
  // 攜帶traceId發(fā)起rpc請求
  repResp := RequestRPC(ctx, ...)
  // 攜帶traceId查詢DB
  dbResp := RequestDB(ctx, ...)
  // ...
}

func RequestRPC(ctx context.Context, ...) interface{} {
    // 獲取traceid,在調(diào)用rpc時記錄日志
    traceId, _ := ctx.Value(TraceIdKey)
    // 發(fā)起請求

    // ...
    return
}

接收到請求后,通過req獲取到traceId并記錄到Context中,在調(diào)用其他RPC服務(wù)和查詢DB時,傳入構(gòu)造的Context。在后續(xù)代碼中,可以通過Context拿到存入的traceId。

責(zé)任編輯:姜華 來源: 今日頭條
相關(guān)推薦

2023-05-12 09:40:53

ContextGolang

2023-11-13 21:55:12

Go編程

2022-02-13 23:10:46

Golang語言變量

2013-12-25 16:03:39

GitGit 命令

2025-04-24 10:40:46

CatalogFlink SQL元數(shù)據(jù)

2024-10-06 12:35:50

2024-04-11 13:41:47

2022-05-07 08:22:34

內(nèi)核BPF代碼

2020-02-12 14:42:00

GPU技術(shù)關(guān)鍵參數(shù)應(yīng)用場景

2019-12-30 10:40:31

GPU技術(shù)應(yīng)用

2023-08-03 08:48:07

Golang接口

2023-11-27 15:02:37

BytesGolang

2021-04-28 09:02:48

Golang語言Context

2023-05-16 07:47:18

RabbitMQ消息隊列系統(tǒng)

2023-10-24 16:03:34

GoGolang

2023-08-02 09:07:27

Golangio 包

2023-10-18 08:22:38

BufioGolang

2024-11-27 08:15:50

2021-04-21 09:21:07

zookeeper集群源碼

2024-01-30 09:43:43

Java緩存技術(shù)
點贊
收藏

51CTO技術(shù)棧公眾號