dqrl3595 2017-03-18 04:37
浏览 22


Go has come with a new package called context and with recent versions (Go 1.7 I think) we should be able to use it in the same way as gorilla/context package:


With gorilla context you can very easily set and get variables that are relevant to a request, it's handlers, and middlewares.

To set a value in gorilla context is really easy:

func handleFunc(w http.ResponseWriter, r *http.Request) {
    context.Set(r, "foo", "bar")

To get the value we can do:

func handleFunc(w http.ResponseWriter, r *http.Request) {
    val := context.Get(r, "foo")

I understand that we can use this in middlewares so that the next middleware can use variables that were set in previous middleware. I would like to be able to do this with Go context package.

I understand that to get a value is quite simple like this:

func handleFunc(w http.ResponseWriter, r *http.Request) {

But I have no idea how to set the value. It was not very intuitive for me and I don't really understand how to do it.

  • 写回答

1条回答 默认 最新

  • douye2111 2017-03-18 05:15

    See "Exploring the context package", using WithValue and the context associated to the Request:


    Middleware in Go refers to an http handler which wraps around a multiplexer. There are several 3rd party middleware solutions (such as negroni), but really the standard library supports a very similar pattern. The use of a Context in the request allows us to hold data in the request.

    See the example code for invocation and definition.

    func putClientIPIntoContext(r *http.Request) context.Context {
        ci := r.RemoteAddr
        fwd := r.Header.Get("X-Forwarded-For")
        if fwd != "" {
            ci = fwd
        ctx := context.WithValue(r.Context(), ClientIPKey, ci)
        return ctx

    The Context can store request-scoped variables.
    It’s useful when writing ‘middleware’, but it’s a little bit ‘anti-pattern’ — it’s a bit magical, because it’s not type-safe.

    See more at "Pitfalls of context values and how to avoid or mitigate them in Go".

    The example below only shows how you might use the authentication logic from above to verify that when a user is logged in when visiting any page with a path prefix of /dashboard/.
    A similar approach could be used to verify that a user is an admin before allowing them access to any page with a path prefix of /admin/.

    func requireUser(next http.Handler) http.Handler {  
      return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
        user := lookupUser(r)
        if user == nil {
          // No user so redirect to login
          http.Redirect(w, r, "/login", http.StatusFound)
        ctx := context.WithValue(r.Context(), "user", user)
        next.ServeHTTP(w, r.WithContext(ctx))
    func main() {  
      dashboard := http.NewServeMux()
      dashboard.HandleFunc("/dashboard/hi", printHi)
      dashboard.HandleFunc("/dashboard/bye", printBye)
      mux := http.NewServeMux()
      // ALL routes that start with /dashboard/ require that a 
      // user is authenticated using the requireUser middleware
      mux.Handle("/dashboard/", requireUser(dashboard))
      mux.HandleFunc("/", home)
      http.ListenAndServe(":3000", addRequestID(mux))

    As kostix comments, use Context wisely, like Dave Cheney suggest in "Context is for cancelation"

    本回答被题主选为最佳回答 , 对您是否有帮助呢?



  • ¥15 Qt4代码实现下面的界面
  • ¥15 prism提示我reinstall prism 如何解决
  • ¥15 asp.core 权限控制怎么做,需要控制到每个方法
  • ¥20 while循环中OLED显示中断中的数据不正确
  • ¥15 这个视频里的stm32f4代码是怎么写的
  • ¥15 JNA调用DLL报堆栈溢出错误(0xC00000FD)
  • ¥15 请教SGeMs软件的使用
  • ¥15 自己用vb.net编写了一个dll文件,如何只给授权的用户使用这个dll文件进行打包编译,未授权用户不能进行打包编译操作?
  • ¥50 深度学习运行代码直接中断
  • ¥20 需要完整的共散射点成像代码