go的ctx超时控制

正确写法

对于要控制超时的函数,要异步调用它,才能做到超时控制

go 复制代码
func TestCtxCancal(t *testing.T) {
	tag := "TestCtxCancal"
	ctx := context.Background()
	// kafka关闭可能耗时较长,设置强制退出时间,设置5s超时
	ctx, cancel := context.WithTimeout(ctx, 5*time.Second)
	defer cancel()
	stop := make(chan struct{}, 1)
	go func() {
		// mock kafka close 耗时很长,耗时15s
		time.Sleep(time.Second * 15)
		logger.Infow(ctx, tag, "msg", "kafka has closed")
		stop <- struct{}{}
	}()

	hasCancelTimeout := false
	select {
	case <-ctx.Done():
		logger.Error(context.TODO(), "sercer.main", "kafka Shutdown timed out, might not have stopped gracefully")
		hasCancelTimeout = true
	case <-stop:
		logger.Info(context.TODO(), "main", "kafka consume has stopped")
		hasCancelTimeout = false
	}

    //校验代码进入了select的超时case 
	assert.True(t, hasCancelTimeout)
}

错误写法

同步调用要控制超时的函数,就会导致必须等到函数执行完后,才走到select的代码,超时控制失去作用

go 复制代码
func TestCtxCancalFail(t *testing.T) {
	tag := "TestCtxCancal"
	ctx := context.Background()
	// kafka关闭可能耗时较长,设置强制退出时间
	timeout := 5 * time.Second
	ctx, cancel := context.WithTimeout(ctx, timeout)
	defer cancel()
	stop := make(chan struct{}, 1)

	start := time.Now()
	// 如果同步调用close,那么监听超时代码就失去意义
	{
		// mock kafka close 耗时很长
		time.Sleep(time.Second * 15)
		logger.Infow(ctx, tag, "msg", "kafka has closed")
		stop <- struct{}{}
	}

	select {
	case <-ctx.Done():
		logger.Error(context.TODO(), "sercer.main", "kafka Shutdown timed out, might not have stopped gracefully")
	case <-stop:
		logger.Info(context.TODO(), "main", "kafka consume has stopped")
	}

	elapsed := time.Since(start)
	// 证明同步调用要控制超时的函数,cancel的超时控制就失去意义了
	// 10s已经远大于 timeout := 5 * time.Second
	assert.True(t, elapsed > 10*time.Second)
}

总结

go的ctx超时控制常用于服务的优雅退出,在释放资源时,加一个超时控制,避免服务迟迟未能退出。在使用context.WithTimeout时,要注意正确异步调用要控制超时的函数

相关推荐
用户3983461612026 分钟前
Go-Spring 实战第 10 课 —— 依赖注入的方式:字段注入和构造函数注入
spring·go
用户398346161203 小时前
Go-Spring 实战第 9 课 —— IoC 容器:复杂 Go 应用如何统一对象装配
spring·go
审判长烧鸡3 小时前
【Go Generics】泛型为何而生的
go·泛型·overload·重载·generics
用户398346161204 小时前
Go-Spring 实战第 8 课 —— 变量引用与动态刷新:配置值如何复用和更新
spring·go
小羊在睡觉1 天前
力扣239. 滑动窗口最大值
数据结构·后端·算法·leetcode·go
用户398346161201 天前
Go-Spring 实战第 7 课 —— Profile 多环境配置:基础配置与环境差异如何避免复制
spring·go
审判长烧鸡1 天前
【Go Context】终极指南
go
审判长烧鸡1 天前
【Go Test】单元测试保姆级完整指南
单元测试·go
审判长烧鸡2 天前
【Go工具】go-playground是什么组织?官方的?
开发语言·安全·go
别样的感动2 天前
我写了一个 Go 框架:用 DSL 替代 ORM,代码体积减半,开发效率翻倍
go