Go学习笔记—标准库Context

标准库Context

​ 由于goroutine没有父子关系,多个goroutine都是被平行的调度,所以在拉起多个goroutine后,程序的执行模型并没有维护树状结构的goroutine树,所以无法靠语法层面,通知树中所有goroutine退出。context库就是为了解决这个问题而被引入,目的:

(1) 退出通知机制:通知可以传递给整个goroutine调用树上的每一个goroutine

(2) 传递数据:数据可以传递给goroutine调用树上的每一个goroutine

1. context—基础数据结构

——构造取消树根节点对象的两个函数

func Background() Context构造根节点,用作WithConcel的实参
func Todo() Context

——构建不同功能的Context对象

​ 这些方法都有一个参数parent,即在goroutine调用链中,每层都对Context实例包装自己所需的功能,并传递到下一层。

func WithCancel(parent Context) (ctx Context, cancel CancelFunc)带有退出通知的Context对象
func WithDeadline(parent Context, deadline time.Time) (Context, CancelFunc)带有超时通知的Context对象
func WithTimeout(parent Context, timeout time.Duration) (Context, CancelFunc)带有超时通知的Context对象
func WithValue(parent Context, key, value interface{}) Context能够传递数据的Context对象
2. context—用例

ctxa.children——>ctxb、ctxb.children——>ctxc

  • ctxa 创建时:&cancelCtx{Context: new(emptyCtx)}
  • ctxb 创建时: &timerCtx{cancelCtx: ctxa deadline:tm} 同时出发ctxa,在children中加入ctxb
  • ctxc 创建时:&cancelCtx{Context: mc} 同时通过mc.Context找到ctxb,通过ctxb.cancelCtx找到ctxa,在ctxa的children中加入ctxc
type selfcontext struct {
	context.Context
}

func work(ctx context.Context, name string) { //一个子任务,参数包含上下文信息与子任务名
	for {
		select {
		case <-ctx.Done(): //从上下文接收到结束信息
			fmt.Printf("%s get msg to cancel\n", name)
			return
		default:
			fmt.Printf("%s is running\n", name)
			time.Sleep(1 * time.Second)
		}
	}
}

func workWithValue(ctx context.Context, name string) { //一个子任务,参数包含带值的上下文信息与子任务名
	for {
		select {
		case <-ctx.Done(): //从上下文接收到结束信息
			fmt.Printf("%s get msg to cancel\n", name)
			return
		default:
			value := ctx.Value("key").(string) //获取上下文中存储的key值
			fmt.Printf("%s is running and value is %s\n", name, value)
			time.Sleep(1 * time.Second)
		}
	}
}

func ContextTest() {
	//使用background构建一个WithCancel类型的上下文
	ctxa, cancel := context.WithCancel(context.Background())
	go work(ctxa, "work1") //在此上下文,开启一个goroutine
	//使用ctxa构建一个WithDeadline类型的上下文
	tm := time.Now().Add(3 * time.Second)
	ctxb, _ := context.WithDeadline(ctxa, tm)
	go work(ctxb, "work2") //在此上下文,开启一个goroutine
	//使用ctxb构建一个WithValue类型的上下文
	mc := selfcontext{ctxb}
	ctxc := context.WithValue(mc, "key", "i am work3")
	go workWithValue(ctxc, "work3") //在此上下文,开启一个goroutine
	time.Sleep(10 * time.Second) //经过10s,由于ctxc,ctxb都包含Deadline上下文,所以都已经超时停止
	//显示调用wordk1的cancal()方法,结束work1
	cancel()
}

总结流程如下:

  1. 创建一个Context根对象 (通过Background或TODO创建)
  2. 包装上一步的Context对象,并加入特有功能 (通过WithCancel,WithTimeout,WithDeadline,WithValue创建)
  3. 将上一步创建的对象作为实参传递给后续启动的并发函数(一般作为其第一个参数),每个并发函数可以继续使用包装函数对传进来的Context对象进行包装,添加自己所需要的功能。
  4. 顶端的goroutine在超时后调用cancel退出通知函数,通知后面的所有goroutine释放资源。
  5. 后面的goroutine通过select监听Context.Done()返回的chan,及时相应前端goroutine的退出通知,释放资源。
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
Go语言学习笔记.pdf》是一本关于Go语言学习学习笔记,内容丰富且简洁明了。本书从基础知识开始,逐步介绍了Go语言的语法、特性和常用函数等。在学习笔记中,作者通过实际的示例和练习帮助读者理解Go语言的概念和用法。 第一章介绍了Go语言的起源和发展,为读者提供了对Go语言背景的整体了解。第二章讲解了Go语言的基本语法,例如变量声明、循环和条件语句等。通过大量的代码示例,读者能够更好地理解Go语言的语法和结构。 接下来的章节重点介绍了Go语言的并发编程和高级特性。第三章详细介绍了Go语言中的goroutine和channel,这是Go语言并发编程的核心机制。作者通过生动的示例代码和实际应用案例,向读者展示了如何使用goroutine和channel实现并发编程。 第四章和第五章分别介绍了Go语言中的面向对象编程和函数式编程。通过深入讲解Go语言中的结构体、接口和函数,读者能够更好地应用这些特性进行代码设计和开发。 最后几章则介绍了Go语言中常用的函数和工具。例如,第六章介绍了Go语言中用于网络编程的net包和http包。读者可以学习到如何使用这些函数构建基于网络的应用程序。 总的来说,《Go语言学习笔记.pdf》是一本非常实用的Go语言学习资料。通过阅读这本书,读者能够系统地学习和理解Go语言的基本概念和高级特性,为之后的Go语言开发打下坚实的基础。无论是初学者还是有一定编程经验的开发者,都能从中获得丰富的知识和经验。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Ostrich5yw

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值