在PHP中,协程是一种强大的编程工具,可以提高代码的执行效率。而在协程中,将主协程上下文的副本传递给子例程上下文是一种常用的操作。通过这种方式,可以在子例程中访问主协程的上下文数据,实现数据的共享和传递。这一过程在PHP中非常简单,只需要使用yield关键字即可实现。在本文中,我们将由php小编小新为您详细介绍如何将主协程上下文的副本传递给子例程上下文,并给出一些实例代码,帮助您更好地理解和应用这一特性。
我有一个 golang API 端点及其关联的上下文。
端点需要在幕后执行一些繁重的工作,因此我在主端点内创建一个新的子例程,然后返回响应本身。
为了处理上下文取消,我创建了一个后台上下文并将其作为新上下文传递给子例程。
问题是通过这样做,是的,我可以执行后台子例程,但是主上下文中的值,例如请求ID,跨度ID等(大多数键我都不知道),这些值被用于追踪将会丢失。
即使在响应发送到客户端后,如何将父上下文传递给子例程而不取消执行。
我没有将任何值传递到上下文中。 但最初我们传递的是跟踪所需的 request-id、span-id 等。 这些信息都在上下文中。 这是一个内部库,上下文是我们保存它的地方。
我知道这是使用上下文传递值的反模式,除了请求 ID 和其他对库而不是业务逻辑重要的值之外,不会传递任何值
当您取消父上下文时,从其派生的所有上下文也将取消。因此,您为请求处理程序生成的 goroutine 创建新上下文是正确的。
当您创建新上下文时,您应该将您感兴趣的所有值从原始上下文复制到新上下文。然而,你说你不知道所有的钥匙。因此,您仍然可以保留对父上下文的引用,以便可以查询它的值。像这样的事情:
type nestedContext struct { context.Context parent context.Context } func (n nestedContext) Value(key any) any { return n.parent.Value(key) } ... newContext := nestedContext{ Context:context.Background(), parent: parentContext, }
这将从 context.background()
创建一个新上下文,该上下文将从取消的父上下文中查找值。
将 newcontext
作为上下文传递给从处理程序创建的 goroutine。
以上是将主协程上下文的副本传递给子例程上下文的详细内容。更多信息请关注PHP中文网其他相关文章!