首页 > 后端开发 > Golang > 正文

Golang使用context控制并发任务生命周期

P粉602998670
发布: 2025-09-04 08:17:01
原创
461人浏览过
使用context.WithCancel可取消goroutine,调用cancel()后所有监听该context的goroutine通过ctx.Done()收到信号并退出。

golang使用context控制并发任务生命周期

在Golang中,

context
登录后复制
包提供了一种优雅的方式来管理并发任务的生命周期,包括取消、超时和传递请求相关的值。它允许你构建更健壮、更可控的并发程序。

使用

context
登录后复制
可以控制并发任务的启动、停止和传递请求范围的数据。

如何使用Context取消一个goroutine?

context.WithCancel
登录后复制
函数是实现取消goroutine的关键。它返回一个
context
登录后复制
和一个
CancelFunc
登录后复制
。调用
CancelFunc
登录后复制
会取消
context
登录后复制
,所有监听该
context
登录后复制
的goroutine都会收到取消信号。

package main

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

func worker(ctx context.Context, id int) {
    for {
        select {
        case <-ctx.Done():
            fmt.Printf("Worker %d: 任务取消\n", id)
            return
        default:
            fmt.Printf("Worker %d: 正在工作中...\n", id)
            time.Sleep(time.Second)
        }
    }
}

func main() {
    ctx, cancel := context.WithCancel(context.Background())

    // 启动多个worker goroutine
    for i := 1; i <= 3; i++ {
        go worker(ctx, i)
    }

    // 模拟一段时间后取消任务
    time.Sleep(3 * time.Second)
    fmt.Println("准备取消所有worker...")
    cancel()

    // 等待一段时间,确保所有worker都已退出
    time.Sleep(time.Second)
    fmt.Println("所有worker已退出,程序结束")
}
登录后复制

在这个例子中,我们创建了一个带有取消功能的

context
登录后复制
。启动了三个
worker
登录后复制
goroutine,每个goroutine都在循环中执行任务,并监听
context.Done()
登录后复制
通道。当调用
cancel()
登录后复制
函数时,
context
登录后复制
被取消,所有
worker
登录后复制
goroutine收到信号并退出。 我个人觉得这种方式比直接使用channel发送信号更加优雅,尤其是当你需要传递多个取消信号的时候。

立即学习go语言免费学习笔记(深入)”;

Context的超时控制是如何实现的?

context.WithTimeout
登录后复制
函数允许你设置一个超时时间。当超过指定时间后,
context
登录后复制
会自动取消。这对于防止goroutine无限期地阻塞非常有用。

package main

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

func longRunningTask(ctx context.Context) {
    select {
    case <-time.After(5 * time.Second):
        fmt.Println("任务完成")
    case <-ctx.Done():
        fmt.Println("任务超时取消")
    }
}

func main() {
    ctx, cancel := context.WithTimeout(context.Background(), 3*time.Second)
    defer cancel() // 确保即使任务提前完成,cancel也会被调用

    go longRunningTask(ctx)

    select {
    case <-ctx.Done():
        fmt.Println("主程序检测到任务超时")
    case <-time.After(6 * time.Second): // 稍微长于longRunningTask,确保其完成或超时
        fmt.Println("主程序结束")
    }
}
登录后复制

在这个例子中,

longRunningTask
登录后复制
模拟一个需要5秒才能完成的任务。我们使用
context.WithTimeout
登录后复制
设置了3秒的超时时间。如果任务在3秒内没有完成,
context
登录后复制
将被取消,
longRunningTask
登录后复制
会收到取消信号并退出。 注意
defer cancel()
登录后复制
,这是一个好习惯,即使任务提前完成,也应该调用
cancel
登录后复制
释放资源。

GPT-MINUS1
GPT-MINUS1

通过在文本中随机地用同义词替换单词来愚弄GPT

GPT-MINUS1 83
查看详情 GPT-MINUS1

如何在Context中传递请求相关的值?

context.WithValue
登录后复制
函数允许你在
context
登录后复制
中存储键值对。这些值可以在goroutine之间传递,例如请求ID、用户信息等。

package main

import (
    "context"
    "fmt"
)

func processRequest(ctx context.Context) {
    userID := ctx.Value("userID")
    fmt.Printf("处理请求,用户ID: %v\n", userID)
}

func main() {
    ctx := context.WithValue(context.Background(), "userID", "12345")
    processRequest(ctx)
}
登录后复制

在这个例子中,我们在

context
登录后复制
中存储了
userID
登录后复制
processRequest
登录后复制
函数可以从
context
登录后复制
中获取
userID
登录后复制
并使用它。 需要注意的是,
context.Value
登录后复制
的键应该是可比较的类型,通常是自定义类型,以避免与其他包冲突。 此外,不要在
context
登录后复制
中传递可选参数,应该使用函数参数来传递。

Context的父子关系是如何工作的?

context
登录后复制
可以有父子关系。子
context
登录后复制
继承父
context
登录后复制
的值、截止时间和取消信号。如果父
context
登录后复制
被取消,所有子
context
登录后复制
也会被取消。

package main

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

func childTask(ctx context.Context) {
    select {
    case <-ctx.Done():
        fmt.Println("子任务被取消")
    case <-time.After(2 * time.Second):
        fmt.Println("子任务完成")
    }
}

func main() {
    ctx, cancel := context.WithCancel(context.Background())

    go childTask(ctx)

    time.Sleep(1 * time.Second)
    fmt.Println("取消父任务")
    cancel()

    time.Sleep(1 * time.Second) // 等待子任务退出
}
登录后复制

在这个例子中,

childTask
登录后复制
context
登录后复制
是父
context
登录后复制
的子
context
登录后复制
。当父
context
登录后复制
被取消时,
childTask
登录后复制
也会收到取消信号。 这种父子关系使得你可以构建复杂的并发任务树,并统一管理它们的生命周期。

Context的最佳实践是什么?

  • 始终传递
    context
    登录后复制
    作为函数的第一个参数,尤其是当函数会启动goroutine时。
  • 使用
    context.TODO
    登录后复制
    作为顶级
    context
    登录后复制
    的初始值,表示你还不清楚需要传递什么值。
  • 不要在
    context
    登录后复制
    中存储必需的参数,应该使用函数参数来传递。
  • context
    登录后复制
    是不可变的,不要修改它。
  • 确保在任务完成后调用
    cancel
    登录后复制
    函数,释放资源。
  • 使用自定义类型作为
    context.Value
    登录后复制
    的键,避免与其他包冲突。
  • 理解
    context
    登录后复制
    的取消和超时机制,并合理地使用它们来管理goroutine的生命周期。

总而言之,

context
登录后复制
是Golang并发编程中一个非常强大的工具。 掌握
context
登录后复制
的使用,可以帮助你构建更健壮、更可控的并发程序。

以上就是Golang使用context控制并发任务生命周期的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号