7.13. 函数文本和闭包

处理函数(handler)中捕捉错误是一些类似的重复代码。如果我们想将捕捉错误的代码封装成一个函数,应该怎么做?GO的函数文本提供了强大的抽象能力,可以帮我们做到这点。

首先,我们重写每个处理函数的定义,让它们接受标题字符串:

定义一个封装函数,接受上面定义的函数类型,返回http.HandlerFunc(可以传送给函数http.HandleFunc)。

  func makeHandler(fn func (http.ResponseWriter, *http.Request, string)) http.HandlerFunc {
      return func(w http.ResponseWriter, r *http.Request) {
          // Here we will extract the page title from the Request,
          // and call the provided handler 'fn'
      }
  }

返回的函数称为闭包,因为它包含了定义在它外面的值。在这里,变量fn(makeHandler的唯一参数)被闭包包含。fn是我们的处理函数,save、edit、或view。

我们可以把getTitle的代码复制到这里(有一些小的变动):

  func makeHandler(fn func(http.ResponseWriter, *http.Request, string)) http.HandlerFunc {
      return func(w http.ResponseWriter, r *http.Request) {
          title := r.URL.Path[lenPath:]
          if !titleValidator.MatchString(title) {
              http.NotFound(w, r)
              return
          }
          fn(w, r, title)
      }
  }

makeHandler返回的闭包是一个函数,它有两个参数,http.Conn和http.Request(因此,它是http.HandlerFunc)。闭包从请求路径解析title,使用titleValidator验证标题。如果title无效,使用函数http.NotFound将错误写到Conn。如果title有效,封装的处理函数fn将被调用,参数为Conn, Request, 和title。

在main函数中,我们用makeHandler封装所有处理函数:

  func main() {
      http.HandleFunc("/view/", makeHandler(viewHandler))
      http.HandleFunc("/edit/", makeHandler(editHandler))
      http.HandleFunc("/save/", makeHandler(saveHandler))
      http.ListenAndServe(":8080", nil)
  }

最后,我们可以删除处理函数中的getTitle,让处理函数更简单。

  func viewHandler(w http.ResponseWriter, r *http.Request, title string) {
      p, err := loadPage(title)
      if err != nil {
          http.Redirect(w, r, "/edit/"+title, http.StatusFound)
          return
      }
      renderTemplate(w, "view", p)
  }

  func editHandler(w http.ResponseWriter, r *http.Request, title string) {
      p, err := loadPage(title)
      if err != nil {
          p = &page{title: title}
      }
      renderTemplate(w, "edit", p)
  }

  func saveHandler(w http.ResponseWriter, r *http.Request, title string) {
      body := r.FormValue("body")
      p := &page{title: title, body: []byte(body)}
      err := p.save()
      if err != nil {
          http.Error(w, err.String(), http.StatusInternalServerError)
          return
      }
      http.Redirect(w, r, "/view/"+title, http.StatusFound)
  }