为什么在处理程序中的goroutine中要复制上下文?

huangapple go评论100阅读模式
英文:

Why should I make a copy of a context for goroutines inside handlers?

问题

最近我开始用Go重写一些Python服务,以提高它们的速度,并在gin文档中找到了这一部分内容:https://github.com/gin-gonic/gin#goroutines-inside-a-middleware

所以我理解了这些说明,但我试图理解为什么要这样做?复制上下文的重要性是什么?如果在处理程序中的goroutine中不复制上下文会引入什么问题?

英文:

I recently started rewriting some of my Python services in Go to speed them up and came across this section of the gin documentation:
https://github.com/gin-gonic/gin#goroutines-inside-a-middleware

So I understand the instructions, but I'm trying to understand why? What is the significance of making a copy, and what problem is introduced if I do not make a copy of context for goroutines within handlers?

答案1

得分: 1

Gin-gonic是自身异步的,这使得它非常出色。如果在处理程序中使用并发,很可能会遇到以下情况:

  • Context是一个结构体,它保存了每个请求的数据(参数、由互斥锁保护的键),因此可能会过时。
  • Context为空,这将导致回退到默认的Context。

下面是Context的结构:

  1. type Context struct {
  2. writermem responseWriter
  3. Request *http.Request
  4. Writer ResponseWriter
  5. Params Params
  6. handlers HandlersChain
  7. index int8
  8. fullPath string
  9. engine *Engine
  10. params *Params
  11. skippedNodes *[]skippedNode
  12. // This mutex protects Keys map.
  13. mu sync.RWMutex
  14. // Keys is a key/value pair exclusively for the context of each request.
  15. Keys map[string]any
  16. // Errors is a list of errors attached to all the handlers/middlewares who used this context.
  17. Errors errorMsgs
  18. // Accepted defines a list of manually accepted formats for content negotiation.
  19. Accepted []string
  20. // queryCache caches the query result from c.Request.URL.Query().
  21. queryCache url.Values
  22. // formCache caches c.Request.PostForm, which contains the parsed form data from POST, PATCH,
  23. // or PUT body parameters.
  24. formCache url.Values
  25. // SameSite allows a server to define a cookie attribute making it impossible for
  26. // the browser to send this cookie along with cross-site requests.
  27. sameSite http.SameSite
  28. }

为了预见这些问题和竞态条件,你必须在处理程序使用并发时使用Copy

以下是gin-gonic存储库中的一句引用:

  1. // Copy returns a copy of the current context that can be safely used outside the request's scope.
  2. // This has to be used when the context has to be passed to a goroutine.

这段代码返回一个当前上下文的副本,可以安全地在请求范围之外使用。当上下文必须传递给goroutine时,必须使用它。

英文:

Gin-gonic is async itself which makes it great.
If you are using concurrency within your handlers it's very likely you will face situation when Context, which is struct goes

  • outdated since it's holding the data from each request (parameters, keys protected by mutex)

  • empty, which will cause a fallback to default Context

here's how it looks like:

  1. type Context struct {
  2. writermem responseWriter
  3. Request *http.Request
  4. Writer ResponseWriter
  5. Params Params
  6. handlers HandlersChain
  7. index int8
  8. fullPath string
  9. engine *Engine
  10. params *Params
  11. skippedNodes *[]skippedNode
  12. // This mutex protects Keys map.
  13. mu sync.RWMutex
  14. // Keys is a key/value pair exclusively for the context of each request.
  15. Keys map[string]any
  16. // Errors is a list of errors attached to all the handlers/middlewares who used this context.
  17. Errors errorMsgs
  18. // Accepted defines a list of manually accepted formats for content negotiation.
  19. Accepted []string
  20. // queryCache caches the query result from c.Request.URL.Query().
  21. queryCache url.Values
  22. // formCache caches c.Request.PostForm, which contains the parsed form data from POST, PATCH,
  23. // or PUT body parameters.
  24. formCache url.Values
  25. // SameSite allows a server to define a cookie attribute making it impossible for
  26. // the browser to send this cookie along with cross-site requests.
  27. sameSite http.SameSite

}

In order to foresee such issues and race conditions - you must use Copy whenever your handlers are using go concurrency

here's a quote from gin-gonic repository:

  1. // Copy returns a copy of the current context that can be safely used outside the request's scope.
  2. // This has to be used when the context has to be passed to a goroutine.

huangapple
  • 本文由 发表于 2022年9月18日 19:55:19
  • 转载请务必保留本文链接:https://go.coder-hub.com/73762584.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定