广告
返回顶部
首页 > 资讯 > 后端开发 > GO >Go语言context上下文管理的使用
  • 187
分享到

Go语言context上下文管理的使用

2024-04-02 19:04:59 187人浏览 安东尼
摘要

目录context 有什么作用传递共享的数据取消 Goroutine防止 goroutine 泄漏context.Value 的查找过程是怎样的context 有什么作用 conte

context 有什么作用

context 主要用来在goroutine 之间传递上下文信息,包括:取消信号、超时时间、截止时间、k-v 等。

Go 常用来写后台服务,通常只需要几行代码,就可以搭建一个 Http server。

在 Go 的 server 里,通常每来一个请求都会启动若干个 goroutine 同时工作:有些去数据库拿数据,有些调用下游接口获取相关数据……

这些 goroutine 需要共享这个请求的基本数据,例如登陆的 token,处理请求的最大超时时间(如果超过此值再返回数据,请求方因为超时接收不到)等等。当请求被取消或是处理时间太长,这有可能是使用者关闭了浏览器或是已经超过了请求方规定的超时时间,请求方直接放弃了这次请求结果。这时,所有正在为这个请求工作的 goroutine 需要快速退出,因为它们的“工作成果”不再被需要了。在相关联的 goroutine 都退出后,系统就可以回收相关的资源。

在Go 里,我们不能直接杀死协程,协程的关闭一般会用 channel+select 方式来控制。但是在某些场景下,例如处理一个请求衍生了很多协程,这些协程之间是相互关联的:需要共享一些全局变量、有共同的 deadline 等,而且可以同时被关闭。再用 channel+select 就会比较麻烦,这时就可以通过 context 来实现。

一句话:context 用来解决 goroutine 之间退出通知元数据传递的功能。

context 使用起来非常方便。源码里对外提供了一个创建根节点 context 的函数:

func Background() Context

background 是一个空的 context, 它不能被取消,没有值,也没有超时时间。 有了根节点 context,又提供了四个函数创建子节点 context:

func WithCancel(parent Context) (ctx Context, cancel CancelFunc)
func WithDeadline(parent Context, deadline time.Time) (Context, CancelFunc)
func WithTimeout(parent Context, timeout time.Duration) (Context, CancelFunc)
func WithValue(parent Context, key, val interface{}) Context

context 会在函数传递间传递。只需要在适当的时间调用 cancel 函数向 goroutines 发出取消信号或者调用 Value 函数取出 context 中的值。

  • 不要将 Context 塞到结构体里。直接将 Context 类型作为函数的第一参数,而且一般都命名为ctx
  • 不要向函数传入一个 nil 的 context,如果你实在不知道传什么,标准库给你准备好了一个 context:todo
  • 不要把本应该作为函数参数的类型塞到 context 中,context 存储的应该是一些共同的数据。例如:登陆的 session、cookie 等。
  • 同一个 context 可能会被传递到多个 goroutine,别担心,context并发安全的。

传递共享的数据

对于 WEB 服务端开发,往往希望将一个请求处理的整个过程串起来,这就非常依赖于 Thread Local(对于 Go 可理解为单个协程所独有) 的变量,而在 Go 语言中并没有这个概念,因此需要在函数调用的时候传递 context

package main

import (
    "context"
    "fmt"
)
func main() {
    ctx := context.Background()
    process(ctx)
    ctx = context.WithValue(ctx, "traceId", "qcrao-2019")
    process(ctx)
}
func process(ctx context.Context) {
    traceId, ok := ctx.Value("traceId").(string)
    if ok {
        fmt.Printf("process over. trace_id=%s\n", traceId)
    } else {
        fmt.Printf("process over. no trace_id\n")
    }
}

运行结果:

process over. no trace_id
process over. trace_id=qcrao-2019

第一次调用 process 函数时,ctx 是一个空的 context,自然取不出来 traceId。第二次,通过 WithValue 函数创建了一个 context,并赋上了 traceId 这个 key,自然就能取出来传入的 value 值。

取消 goroutine

我们先来设想一个场景:打开外卖的订单页,地图上显示外卖小哥的位置,而且是每秒更新 1 次。app 端向后台发起 websocket 连接(现实中可能是轮询)请求后,后台启动一个协程,每隔 1 秒计算 1 次小哥的位置,并发送给端。如果用户退出此页面,则后台需要“取消”此过程,退出 goroutine,系统回收资源。

func PerfORM() {
    for {
        calculatePos()
        sendResult()
        time.Sleep(time.Second)
    }
}

如果需要实现“取消”功能,并且在不了解 context 功能的前提下,可能会这样做:给函数增加一个指针型的 bool 变量,在 for 语句的开始处判断 bool 变量是发由 true 变为 false,如果改变,则退出循环。

上面给出的简单做法,可以实现想要的效果,没有问题,但是并不优雅,并且一旦协程数量多了之后,并且各种嵌套,就会很麻烦。优雅的做法,自然就要用到 context。

func Perform(ctx context.Context) {
    for {
        calculatePos()
        sendResult()
        select {
        case <-ctx.Done():
            // 被取消,直接返回
            return
        case <-time.After(time.Second):
            // block 1 秒钟 
        }
    }
}

主流程可能是这样的:

ctx, cancel := context.WithTimeout(context.Background(), time.Hour)
go Perform(ctx)
// ……
// app 端返回页面,调用cancel 函数
cancel()

注意一个细节,WithTimeout 函数返回的 context 和 cancelFun 是分开的。context 本身并没有取消函数,这样做的原因是取消函数只能由外层函数调用,防止子节点 context 调用取消函数,从而严格控制信息的流向:由父节点 context 流向子节点 context。

防止 goroutine 泄漏

前面那个例子里,goroutine 还是会执行完,最后返回,可能多浪费一些系统资源。这里改编一个 “如果不用 context 取消,goroutine 就会泄漏的例子”

func gen() <-chan int {
    ch := make(chan int)
    go func() {
        var n int
        for {
            ch <- n
            n++
            time.Sleep(time.Second)
        }
    }()
    return ch
}

这是一个可以生成无限整数的协程,但如果我只需要它产生的前 5 个数,那么就会发生 goroutine 泄漏:

func main() {
    for n := range gen() {
        fmt.Println(n)
        if n == 5 {
            break
        }
    }
    // ……
}

当 n == 5 的时候,直接 break 掉。那么 gen 函数的协程就会执行无限循环,永远不会停下来。发生了 goroutine 泄漏。

用 context 改进这个例子:

func gen(ctx context.Context) <-chan int {
    ch := make(chan int)
    go func() {
        var n int
        for {
            select {
            case <-ctx.Done():
                return
            case ch <- n:
                n++
                time.Sleep(time.Second)
            }
        }
    }()
    return ch
}
func main() {
    ctx, cancel := context.WithCancel(context.Background())
    defer cancel() // 避免其他地方忘记 cancel,且重复调用不影响
    for n := range gen(ctx) {
        fmt.Println(n)
        if n == 5 {
            cancel()
            break
        }
    }
    // ……
}

增加一个 context,在 break 前调用 cancel 函数,取消 goroutine。gen 函数在接收到取消信号后,直接退出,系统回收资源。

context.Value 的查找过程是怎样的

链表有点像,只是它的方向相反:Context 指向它的父节点,链表则指向下一个节点。通过 WithValue 函数,可以创建层层的 valueCtx,存储 goroutine 间可以共享的变量。

查找的时候,会向上查找到最后一个挂载的 context 节点,也就是离得比较近的一个父节点 context

到此这篇关于Go语言context上下文管理的使用的文章就介绍到这了,更多相关Go语言context上下文管理内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

您可能感兴趣的文档:

--结束END--

本文标题: Go语言context上下文管理的使用

本文链接: https://www.lsjlt.com/news/141872.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • Go语言context上下文管理的使用
    目录context 有什么作用传递共享的数据取消 goroutine防止 goroutine 泄漏context.Value 的查找过程是怎样的context 有什么作用 conte...
    99+
    2022-11-13
  • Go语言的context上下文管理怎么使用
    这篇文章主要讲解了“Go语言的context上下文管理怎么使用”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Go语言的context上下文管理怎么使用”吧!context 有什么作用cont...
    99+
    2023-06-29
  • Go语言上下文context底层原理
    目录1. context 介绍2. 基本介绍3. 源码分析3.1 Context 接口3.2 emptyCtx3.3 cancelCtx3.4 timerCtx3.5 valueCt...
    99+
    2022-11-13
  • Go语言基础学习之Context的使用详解
    目录前言基本用法Context控制goroutine的生命周期使用 WithValue() 传递数据使用 WithCancel() 取消操作使用 WithDeadline() 设置截...
    99+
    2023-05-19
    Go语言Context使用 Go语言Context用法 Go Context
  • go语言context包功能及操作使用的方法
    本篇内容介绍了“go语言context包功能及操作使用的方法”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!Context包到底是干嘛用的?我...
    99+
    2023-06-30
  • 如何使用Go语言和Redis实现文件上传下载功能
    如何使用Go语言和Redis实现文件上传下载功能简介在现代Web应用开发中,文件上传和下载是常见的功能需求。本文将介绍如何使用Go语言和Redis来实现文件上传和下载功能,并提供具体的代码示例。一、文件上传功能实现文件上传功能是指将客户端的...
    99+
    2023-10-26
    Go语言 redis 文件传输
  • 如何在Linux系统上使用Git来管理Go语言项目?
    在现代软件开发中,版本控制系统是一个必不可少的工具。其中,Git是最受欢迎的版本控制系统之一。它允许开发人员协同工作,管理代码变更,并跟踪项目历史记录。同时,Go语言也是一个非常流行的编程语言,因为它具有高效、简洁和易于学习的特点。在本文中...
    99+
    2023-10-12
    linux git django
  • Go语言包管理工具Godep的用法
    一、为什么要包管理 默认Go的第三方包都是放在Gopath的src目录下,而且这些包都没有版本号的概念,这样的可能会出现一些问题。 举个例子:当A同事将开发完程序之后,此时可能引用了...
    99+
    2022-11-11
  • 如何在Linux环境下利用GO语言开发高效文件管理工具?
    在Linux环境下,文件管理工具是必不可少的,而GO语言的高效性能和强大的并发处理能力,使它成为开发高效文件管理工具的理想语言。在本文中,我们将介绍如何利用GO语言开发一个高效的文件管理工具。 第一步:安装GO语言环境 在开始开发之前,我们...
    99+
    2023-10-02
    开发技术 linux 文件
  • Python上下文管理器怎么使用
    这篇文章主要介绍了Python上下文管理器怎么使用的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Python上下文管理器怎么使用文章都会有所收获,下面我们一起来看看吧。什么是上...
    99+
    2022-10-19
  • Go语言中如何使用NumPy处理文件?
    NumPy是Python中一个强大的数值计算库,它可以用于处理大型的多维数组和矩阵运算。但是,对于Go语言开发者而言,如何在Go语言中使用NumPy呢? 在Go语言中,我们可以使用GoPy来使用Python的库。GoPy是一个用于在Go语...
    99+
    2023-08-05
    apache 文件 numy
  • Go语言和JavaScript: 如何使用API管理对象?
    在现代应用程序中,API(应用程序接口)已经成为了必不可少的一部分。API 允许开发者在不了解内部实现的情况下使用其他应用程序或服务。对于那些需要在不同的语言和平台之间进行交互的开发者来说,API是非常重要的。在本文中,我们将讨论如何使用...
    99+
    2023-11-12
    对象 javascript api
  • GO语言在Linux上使用NPM进行自然语言处理,可行吗?
    随着人工智能技术的不断发展,自然语言处理已经成为了最热门的领域之一。自然语言处理技术可以让计算机更好地理解人类语言,进而实现更加智能化的应用。而在自然语言处理领域,NPM是一款非常常用的工具包,它可以帮助我们更加便捷地进行自然语言处理。那...
    99+
    2023-11-10
    linux 自然语言处理 npm
  • Path索引文件:Go语言中必备的文件管理工具!
    在Go语言中,文件管理是一个非常重要的方面。无论是在开发过程中还是在实际生产环境中,文件管理都是必不可少的。在Go语言中,我们通常使用Path索引文件来进行文件管理。在本文中,我们将深入探讨Path索引文件的使用方法以及其在Go语言中的重...
    99+
    2023-10-05
    path 索引 文件
  • 如何使用Git管理自然语言处理项目:Go语言学习笔记
    Git是一款非常流行的版本控制工具,可用于管理软件项目的代码和文档。在自然语言处理领域,Git也广泛应用于管理各种文本处理和机器学习项目。本文将介绍如何使用Git管理自然语言处理项目,并提供Go语言学习笔记的演示代码。 安装Git 首...
    99+
    2023-07-06
    git 学习笔记 自然语言处理
  • 如何在Linux上使用GO语言处理大数据?
    随着大数据时代的到来,越来越多的企业开始使用大数据来分析和处理他们的业务数据。在这个过程中,一种高效的编程语言和工具是不可或缺的。GO语言就是这样一种编程语言,它具有高效、简洁、并发和高性能等特点,是处理大数据的理想选择。 在本文中,我们...
    99+
    2023-09-09
    linux javascript 大数据
  • 怎么在python中使用上下文管理
    本篇文章为大家展示了怎么在python中使用上下文管理,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。python主要应用领域有哪些1、云计算,典型应用OpenStack。2、WEB前端开发,众多大型...
    99+
    2023-06-14
  • 如何处理Go语言中的并发文件上传问题
    在Go语言中处理并发文件上传问题时,可以使用goroutine来实现并发上传。下面是一个简单的示例代码:```gopackage m...
    99+
    2023-10-09
    Go语言
  • 如何处理Go语言中的并发文件上传问题?
    如何处理Go语言中的并发文件上传问题?随着互联网的发展,文件上传在日常开发中变得越来越常见。而在文件上传的过程中,处理多个文件的并发上传问题成为了一个关键的考虑因素。本文将介绍如何使用Go语言来处理并发文件上传问题,并提供具体的代码示例。一...
    99+
    2023-10-22
    文件上传 Go语言 并发处理
  • 详解Python中的with语句和上下文管理器
    目录一、with语句的使用二、上下文管理器三、小结如果你有阅读源码的习惯,可能会看到一些优秀的代码经常出现带有 with 关键字的语句,它通常用在什么场景呢?对于系统资源如文件、数据...
    99+
    2022-11-11
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作