go 使用context 实现超时 取消

本文介绍了如何在Go中利用context包来实现超时和取消功能,特别是当避免使用for循环时,通过select语句进行精确控制。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

不适用 for 循环的话使用 注释部分select
在 select 中区分 取消和超时的操作

package main

import (
	"bytes"
	"context"
	"fmt"
	"io"
	"os"
	"time"
)

func main() {
	var res = make(chan string)
	var cancelSignal = make(chan string)
	go func() {
		// 捕获 输入窗口的取消信号
		signal := make([]byte, 1)
		signalBuffer := bytes.NewBuffer(signal)
		if _, err := io.Copy(signalBuffer, os.Stdin); err != nil {
			fmt.Println("read err is " + err.Error())
		}
		cancelSignal <- signalBuffer.String()
	}()
	// 创建 超时和 取消的context, cancel:取消函数
	ctx := context.Background()
	ctx, _ = context.WithTimeout(ctx, time.Second * 10)
	ctx, cancel := context.WithCancel(ctx)
	go test(ctx, res)
	/*
	select {
	case result := <- res:
		fmt.Println(result)
	case <- ctx.Done():
		fmt.Println("超时")
		fmt.Println(ctx.Err())
	case signal := <- cancelSignal:
		fmt.Println(signal)
		fmt.Println("取消")
		cancel()
		fmt.Println(ctx.Err())
	}

	 */
loop:
	for {
		select {
		case result := <- res:
			fmt.Println(result)
			break loop
		case <- ctx.Done():
			fmt.Println(ctx.Err())
			break loop
		case <- cancelSignal:
			fmt.Println("取消")
			cancel()
		default:
			time.Sleep(time.Millisecond)
		}
	}
}

func test(ctx context.Context, res chan string) {
	// create new context
	_, _ = context.WithDeadline(ctx, time.Now().Add(time.Second * 20))
	time.Sleep(time.Second * 20)
	res <- "XYY"
}
Go语言中,使用`context`包是实现协程间取消信号、截止时间等信息传递的有效手段。`context`的根实例通常由`context.Background()`提供,而`context.WithCancel()`可以用来创建一个可取消的上下文实例。在HTTP服务中,为了避免耗时操作导致的资源浪费和潜在阻塞,我们可以设置一个超时上下文,一旦超时发生,相关的协程将被取消。 参考资源链接:[Golang context实战:控制HTTP请求超时与协程管理](https://wenku.youkuaiyun.com/doc/3sioqh2awu?spm=1055.2569.3001.10343) 具体到HTTP请求超时控制的实现,首先需要在HTTP处理器中为每个请求创建一个带有超时限制的`context`。可以使用`context.WithTimeout()`或者`context.WithDeadline()`来设置超时或截止时间。当请求超时或截止时间到达时,`ctx.Done()`通道会被关闭,协程通过监听该通道就可以得知取消信号,并执行相应的清理工作。 下面是一个简化的代码示例,展示了如何在HTTP处理函数中实现请求超时控制: ```go func handleRequest(w http.ResponseWriter, r *http.Request) { ctx, cancel := context.WithTimeout(r.Context(), 5*time.Second) // 5秒超时设置 defer cancel() // 确保函数执行完毕后取消上下文 // 启动一个协程来处理请求 go func(ctx context.Context) { // 这里执行耗时的操作 select { case <-ctx.Done(): // 超时取消信号被触发 log.Println( 参考资源链接:[Golang context实战:控制HTTP请求超时与协程管理](https://wenku.youkuaiyun.com/doc/3sioqh2awu?spm=1055.2569.3001.10343)
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值