Golang中的上下文-context包的简介及使用

文章目录

    • 简介
    • context.Background()
    • 上下文取消函数
    • 上下文值传递
    • 建议
    • Reference

简介

Go语言中的context包定义了一个名为Context的类型,它定义并传递截止日期、取消信号和其他请求范围的值,形成一个链式模型。如果我们查看官方文档,它是这样说的:

context包定义了Context类型,它在API边界和进程之间传递截止日期、取消信号和其他请求范围的值。

包中的主要实体是Context本身,它是一个接口。它只有四个方法:

type Context interface {
    Deadline() (deadline time.Time, ok bool)
    Done() <-chan struct{}
    Err() error
    Value(key interface{}) interface{}
}

这里,

  • Deadline:返回应该取消上下文的时间,以及一个布尔值,当没有截止日期时为false
  • Done:返回一个只接收的空结构体通道,它发出上下文应该被取消的信号
  • Err:在完成通道打开时返回nil;否则它返回上下文取消的原因
  • Value:返回与当前上下文的某个键关联的值,如果没有该键的值,则返回nil

与标准库的其他接口相比,Context有许多方法,这些接口通常只有一两个方法。其中三个密切相关:

  • Deadline是取消的时间
  • Done信号上下文完成时
  • Err返回取消的原因

最后一个方法,Value,返回与某个键关联的值。包的其余部分是一系列函数,允许你创建不同类型的上下文。

为什么使用Context

  • 它简化了跨进程或API的截止日期和取消的实现。
  • 它为你的代码扩展做好准备,例如,使用Context会使你的代码清晰易操纵,通过将所有进程以子父关系链式连接,你可以将任何进程绑定/连接在一起。
  • 它易于使用。
  • 它是Goroutine安全的,即你可以在不同的Goroutine上运行相同的上下文而不会泄露。

context.Background()

Background是一个空的上下文,它不会被取消,没有截止日期,也不持有任何值。它主要由主函数用作根上下文或用于测试目的:

package main

import (
	"context"
	"fmt"
)

func main() {
	ctx := context.Background()
	fmt.Println(ctx)
}

输出:

$ go run main.go
context.Background

输出是“context.Background”,它告诉我们这是一个空的Context,它是一个接口,在这个Context接口上,

所有这些数据目前都是nil或空的,因为我们有一个空的Context,即背景上下文,它永远不会被取消,没有截止日期,也没有值。Background通常用于main、init和测试中,以及作为传入请求的顶级上下文。

让我们对所有这些进行fmt.Println检查:

package main

import (
	"context"
	"fmt"
)

func main() {
	ctx := context.Background()
	fmt.Println("ctx.Err() : ", ctx.Err())
	fmt.Println("ctx.Done() : ", ctx.Done())
	fmt.Println("ctx.Value(\"key\") : ", ctx.Value("key"))
	fmt.Print("ctx.Deadline() : ")
	fmt.Print(ctx.Deadline())
}

输出:

$ go run main.go
ctx.Err() :  <nil>
ctx.Done() :  <nil>
ctx.Value("key") :  <nil>
ctx.Deadline() : 0001-01-01 00:00:00 +0000 UTC false

上下文取消函数

Go语言的Context包被用来处理我们的进程或API中的请求流,通过将子上下文与父上下文链接起来,我们可以使用context.WithDeadlinecontext.WithTimeout方法在链中控制截止日期和取消信号。

Context的底层是无法改变的,他在main函数创建,之后传递给其他子函数,比如goroutine,子函数无法改变context,也无法被子进程取消

实例

与done channel不同的是,done是以关闭让其他关闭,而context中的cancel函数则是被调用的

父函数可以取消子函数

如果子函数创建了自己的子函数,也可以把这个context传递下去

那如果子函数想取消自己的子函数呢?

我们可以创建新的context,基于旧的context

我们来举个例子:

func main() {
// 初始化context
	var wg sync.WaitGroup
	ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
	defer cancel()

// 添加数据
	generator := func(data string, stream chan any) {
		for {
			select {
			case <-ctx.Done():
				return
			case stream <- data:
			}
		}
	}
	infiniteApples := make(chan any)
	go generator("apple", infiniteApples)

	infiniteBananas := make(chan any)
	go generator("banana", infiniteBananas)

	infiniteOranges := make(chan any)
	go generator("orange", infiniteOranges)

	wg.Add(1)
	go func1(ctx, &wg, infiniteApples)
	func2 := genericFunc
	func3 := genericFunc
	wg.Add(1)
	go func2(ctx, &wg, infiniteBananas)
	wg.Add(1)
	go func3(ctx, &wg, infiniteOranges)

	wg.Wait()
}

func func1(ctx context.Context, s *sync.WaitGroup, streams <-chan any) {
	defer s.Done()
	var wg sync.WaitGroup

	doWOrks := func(CTX context.Context) {
		defer wg.Done()
		for {
			select {
			case <-CTX.Done():
				return
			case d, ok := <-streams:
				if !ok {
					fmt.Println("stream closed")
					return
				}
				fmt.Println(d)
			}
		}
	}
	// 基于父上下文设置自己的上下文
	newCtx, cancel := context.WithTimeout(ctx, 3*time.Second)
	defer cancel()

// 启动自己的子函数
	for i := 0; i < 3; i++ {
		wg.Add(1)
		go doWOrks(newCtx)
	}
	wg.Wait()
}

// 沿用父函数的context
func genericFunc(ctx context.Context, s *sync.WaitGroup, apples chan any) {
	defer s.Done()
	for {
		select {
		case <-ctx.Done():
			return
		case d, ok := <-apples:
			if !ok {
				fmt.Println("stream closed")
				return
			}
			fmt.Println(d)
		}
	}
}

在输出过程中,我们会发现apple先停止输出,最后5秒到了,另外两个goroutine也停止运行 。

上下文值传递

在Go语言中,context上下文可以通过WithValue函数来传递值。这个函数接受一个父上下文(parent Context)、一个键(key)和一个值(value),返回一个新的上下文(Context),这个新上下文与父上下文相同,但是增加了一个键值对。这个键值对可以在上下文的整个传递链中被检索。

以下是一个使用WithValue传递值的例子:

package main

import (
	"context"
	"fmt"
)

func main() {
	// 创建一个带有值的上下文
	ctx := context.WithValue(context.Background(), "language", "Go")

	// 传递ctx到函数中
	process(ctx)
}

func process(ctx context.Context) {
	// 从ctx中检索值
	if language, ok := ctx.Value("language").(string); ok {
		fmt.Println("Language:", language)
	}
}

在这个例子中,我们创建了一个带有键"language"和值"Go"的上下文。然后,我们将这个上下文传递给了process函数,在这个函数中,我们检索并打印出了这个值。

需要注意的是,context的值应该是请求范围的数据,而不是全局的。它通常用于传递请求相关的元数据,如请求ID、用户身份信息等。此外,由于context是并发安全的,所以它可以在多个goroutine之间安全地传递和使用。

建议

  • 每当你需要使用context.Context时,确保它总是作为第一个参数。
  • 始终使用“ctx”作为变量名,虽然使用其他变量名也可以正常工作,但遵循大多数人的做法是好的,像这样的事情你不需要与众不同。
  • 确保调用取消函数。
  • 不要在方法中使用结构体来添加上下文,始终将其作为参数添加,即context.Context
  • 不要过度使用context.WithValue

Reference

  1. Golang中context包使用场景和示例详解
  2. Golang Context Complete Tutorial with Examples
  3. The Go Blog: Go Concurrency Patterns: Context

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/526956.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

6.10物联网RK3399项目开发实录-驱动开发之SPI接口的使用(wulianjishu666)

嵌入式实战开发例程&#xff0c;珍贵资料&#xff0c;开发必备&#xff1a; 链接&#xff1a;https://pan.baidu.com/s/1149x7q_Yg6Zb3HN6gBBAVA?pwdhs8b SPI 使用 SPI 简介 SPI 是一种高速的&#xff0c;全双工&#xff0c;同步串行通信接口&#xff0c;用于连接微控制器、…

拥有自己的云环境-域名及备案

序 唠叨两句 之前的文章&#xff0c;讲了如何购买一台云服务器&#xff0c;然后购买之后&#xff0c;如何操作云服务器。当买完云服务器之后&#xff0c;我们就可以使用云服务器提供的公网ip&#xff0c;访问到我们的服务器上。但是&#xff0c;这样怎么能体现我们一个老程序…

第十四届蓝桥杯岛屿个数

题目描述&#xff1a; 小蓝得到了一副大小为 MN 的格子地图&#xff0c;可以将其视作一个只包含字符 0&#xff08;代表海水&#xff09;和 1&#xff08;代表陆地&#xff09;的二维数组&#xff0c;地图之外可以视作全部是海水&#xff0c;每个岛屿由在上/下/左/右四个方向上…

使用 AI 生成正则表达式,告别正则烦恼

如果你有处理正则表达式的需求&#xff0c;那么这个网站&#xff08;autoregex.xyz&#xff09;一定要收藏好。 可以根据文字描述生成正则表达式。 默认是从文字到正则&#xff0c;不用选择。 输入框中输入描述&#xff0c;点击 ”GO“ 按钮。 等待一会儿&#xff0c;即可生…

测试开发面经(Flask,轻量级Web框架)

1. Flask的核心特点 a. 轻量级&#xff1a;核心简洁&#xff0c;只提供了基本的功能&#xff0c;其他高级功能可以通过插件或扩展来添加。 b. 灵活性&#xff1a;允许开发者选择适合自己项目的组件和工具&#xff0c;没有强制的项目结构和设计模式。 c. 易于扩展&#xff1a;提…

搭建python编译环境

目录 1.安装依赖包 2.安装失败进行换源 3. 更新系统 通过C 语言调用 Python 代码&#xff0c;需要先安装 libpython3 的 dev 依赖库&#xff08;不同的 ubuntu 版本下&#xff0c; python 版本 可能会有差异&#xff0c; 比如ubuntu 22.04 里是 libpython3.10-dev &#xff09…

javaScript手写专题——实现instanceof/call/apply/bind/new的过程/继承方式

目录 原型链相关 手写instanceof 实现一个_instance方法&#xff0c;判断对象obj是否是target的实例 测试 手写new的过程 实现一个myNew方法&#xff0c;接收一个构造函数以及构造函数的参数&#xff0c;返回构造函数创建的实例对象 测试myNew方法 手写类的继承 ES6&…

人民大学:揭示大语言模型事实召回的关键机制

引言&#xff1a;大语言模型事实召回机制探索 该论文深入研究了基于Transformer的语言模型在零射击和少射击场景下的事实记忆任务机制。模型通过任务特定的注意力头部从语境中提取主题实体&#xff0c;并通过多层感知机回忆所需答案。作者提出了一种新的分析方法&#xff0c;可…

dll文件丢失怎么恢复,教你5种简单有效的方法

在计算机系统的运行过程中&#xff0c;动态链接库&#xff08;DLL&#xff09;文件扮演着至关重要的角色。它们作为共享函数库&#xff0c;封装了大量的可重用代码&#xff0c;使得多个应用程序能够高效调用并执行特定功能&#xff0c;极大地节省了系统资源&#xff0c;提升了软…

Arduino开发 esp32cam+opencv人脸识别距离+语音提醒

效果 低于20厘米语音提醒字体变红 QQ录屏20240406131651 Arduino代码 可直接复制使用&#xff08;修改自己的WIFI) #include <esp32cam.h> #include <WebServer.h> #include <WiFi.h> // 设置要连接的WiFi名称和密码 const char* WIFI_SSID "gumou&q…

指针的深入理解(六)

指针的深入理解&#xff08;六&#xff09; 个人主页&#xff1a;大白的编程日记 感谢遇见&#xff0c;我们一起学习进步&#xff01; 文章目录 指针的深入理解&#xff08;六&#xff09;前言一. sizeof和strlen1.1sizeof1.2strlen1.3sizeof和strlen对比 二.数组名和指针加减…

动态代理

动态代理 动态代理和静态代理角色一致。 代理类是动态生成的,不是我们直接写好的。 动态代理分为俩大类:基于接口的动态代理、基于类的动态代理 基于接口:JDK动态代理(以下示例就是这个) 基于类:cglib java字节码实现:javasist JDK动态代理 InvocationHandler Proxy …

C语言从入门到实战————编译和链接

目录 前言 1. 翻译环境和运行环境 2. 翻译环境 2.1 预处理&#xff08;预编译&#xff09; 2.2 编译 2.2.1 词法分析&#xff1a; 2.2.2 语法分析 2.2.3 语义分析 2.3 汇编 2.4 链接 3. 运行环境 前言 编译和链接是将C语言源代码转换成可执行文件的必经过程&a…

分公司=-部门--组合模式

1.1 分公司不就是一部门吗&#xff1f; "我们公司最近接了一个项目&#xff0c;是为一家在全国许多城市都有分销机构的大公司做办公管理系统&#xff0c;总部有人力资源、财务、运营等部门。" "这是很常见的OA系统&#xff0c;需求分析好的话&#xff0…

Linux 内核移植exfat驱动

简介&#xff1a; Linux系统默认可以自动识别到fat32格式的盘&#xff0c;但fat32支持的文件不能大于4G&#xff0c;所以只能将移动硬盘和U盘格式化为NTFS和exFAT这两种格式的&#xff0c;对于U盘最好格式化为exFAT。 Linux5.4以上的内核原生支持exfat格式&#xff0c;不需要你…

【LeetCode: 572. 另一棵树的子树 + 二叉树 + dfs】

&#x1f680; 算法题 &#x1f680; &#x1f332; 算法刷题专栏 | 面试必备算法 | 面试高频算法 &#x1f340; &#x1f332; 越难的东西,越要努力坚持&#xff0c;因为它具有很高的价值&#xff0c;算法就是这样✨ &#x1f332; 作者简介&#xff1a;硕风和炜&#xff0c;…

UE4_动画基础_ 使用分层动画(Using Layered Animations)

完成在移动过程中武器发射的角色制作&#xff01; 动画混合仅仅意味着在一个角色或骨架网格体上的两个或多个动画之间进行平滑过渡。在虚幻引擎4中&#xff0c;有多种方法可以应用这种混合&#xff0c;要么通过混合空间&#xff0c;或通过实际组合两个基于加权偏差或alpha值的…

开源免费的多功能PDF工具箱

它支持修改PDF、编辑PDF书签、导出PDF书签、导入书签、生成、合并、拆分、提取页面内容、提取图片、OCR 功能介绍: 修改PDF信息&#xff1a;修改文档属性、页码编号、页面链接、页面尺寸&#xff1b;删除自动打开网页等动作&#xff0c;去除复制及打印限制&#xff1b;设置阅读…

SpringBoot中这样用ObjectMapper,才够优雅!

目录 背景步骤在SpringBoot项目中要实现对象与Json字符串的互转&#xff0c;每次都需要像如下一样new 一个ObjectMapper对象&#xff1a;这样的代码到处可见&#xff0c;有问题吗&#xff1f;我们要使用jmh测试几种方式的区别&#xff1a;所以在我们真正使用的时候不要在方法中…

tesseract-ocr一站式安装与使用

目录 前言 安装tesseract-ocr 添加环境变量 1、在path中添加 2、在系統變量中添加 3、验证是否添加成功 添加语言包 更多语言包下载 示例程序 前言 如果你遇到了&#xff1a;make sure the TESSDATA_PREFIX Failed loading language \‘chi_sim 那么就是语言包缺少这个&#xf…