ServeHTTP如何运作?

Dan*_*T29 6 go web-development-server

我正在Golang学习Web开发(初学者)我遇到了一些我玩过的代码,我不太清楚它为什么会起作用,我查看了库的源代码和文档,但我只是有一个模糊的想法,它仍然不是'点击.请注意以下代码:

package main

import (
    "fmt"
    "net/http"
)

type foo int

func (m foo) ServeHTTP(w http.ResponseWriter, r *http.Request) {
    fmt.Fprintln(w, "Some text")
}

func main() {
    var bar foo
    http.ListenAndServe(":8080", bar)
}
Run Code Online (Sandbox Code Playgroud)

根据我的理解,添加ServeHTTP(w http.ResponseWriter,r*http.Request)作为函数方法,调用处理程序接口(如果我说的正确),现在foo也是类型处理程序.我也明白http.ListenAndServe接受类型处理程序的输入,这是我的变量发挥作用的地方.当我运行代码并在浏览器上转到localhost:8080时,我会看到"Some Text".

编辑: 实现接口是正确的术语NOT调用.

题:

这究竟如何运作?如何访问ServeHTTP功能?

我尝试查看库的源代码,但无法准确确定ServeHTTP的工作原理.我找到了这两段代码(不确定这是否适用),这让我觉得它正在实现一个功能,但需要澄清:

// The HandlerFunc type is an adapter to allow the use of
// ordinary functions as HTTP handlers. If f is a function
// with the appropriate signature, HandlerFunc(f) is a
// Handler that calls f.
type HandlerFunc func(ResponseWriter, *Request)

// ServeHTTP calls f(w, r).
func (f HandlerFunc) ServeHTTP(w ResponseWriter, r *Request) {
    f(w, r)
}
Run Code Online (Sandbox Code Playgroud)

我从未见过类型声明,因为HandlerFunc在类型名称后面有一个函数.我还看到了如何声明方法,但不确定上面的代码中发生了什么.

Hen*_*ski 14

这究竟如何运作?如何访问ServeHTTP功能?

要回答这个问题,我们需要看看它是如何http.ListenAndServe工作的:

func ListenAndServe(addr string, handler Handler) error {
    server := &Server{Addr: addr, Handler: handler}
    return server.ListenAndServe()
}
Run Code Online (Sandbox Code Playgroud)

在这里,我们创建一个具有给定地址和处理程序的服务器,并调用ListenAndServer方法,让我们来看看:

func (srv *Server) ListenAndServe() error {
    addr := srv.Addr
    if addr == "" {
        addr = ":http"
    }
    ln, err := net.Listen("tcp", addr)
    if err != nil {
        return err
    }
    return srv.Serve(tcpKeepAliveListener{ln.(*net.TCPListener)})
}
Run Code Online (Sandbox Code Playgroud)

这个方法只是开始监听给定的地址,并用我们新创建的监听器调用Server方法,所以让我们按照那里的路径:

func (srv *Server) Serve(l net.Listener) error {
    defer l.Close()

    ...

    for {
        rw, e := l.Accept()

        ...

        c := srv.newConn(rw)
        c.setState(c.rwc, StateNew) // before Serve can return
        go c.serve(ctx)
    }
}
Run Code Online (Sandbox Code Playgroud)

从Serve方法我们可以看出,这是我们接受新连接并开始在它自己的goroutine中处理它的点.

// Serve a new connection.
func (c *conn) serve(ctx context.Context) {
    ...
    for {
        w, err := c.readRequest(ctx)
        ...
        serverHandler{c.server}.ServeHTTP(w, w.req)
        ...
    }
}
Run Code Online (Sandbox Code Playgroud)

在这里,我们最终调用了ServeHTTP方法,但正如我们所看到的,这还不是我们对该函数的实现,而是来自标准库的内容,所以让我们来看看serverHandler结构包含的内容:

// serverHandler delegates to either the server's Handler or
// DefaultServeMux and also handles "OPTIONS *" requests.
type serverHandler struct {
    srv *Server
}

func (sh serverHandler) ServeHTTP(rw ResponseWriter, req *Request) {
    handler := sh.srv.Handler
    if handler == nil {
        handler = DefaultServeMux
    }
    if req.RequestURI == "*" && req.Method == "OPTIONS" {
        handler = globalOptionsHandler{}
    }
    handler.ServeHTTP(rw, req)
}
Run Code Online (Sandbox Code Playgroud)

所以这里终于来了.如果我们没有提供任何Handler,将使用DefaultServeMux但是因为我们提供了来自foo get的foo处理程序ServeHTTP.

就是这样.所有这些都可以从server.go找到


Adr*_*ian 6

Go 的 HTTP 服务器接收一个地址和一个处理程序。在内部,它创建一个 TCP 侦听器来接受给定地址上的连接,并且每当请求进来时,它:

  1. 将原始 HTTP 请求(路径、标头等)解析为 http.Request
  2. 创建一个http.ResponseWriter用于发送响应
  3. 通过调用处理程序的ServeHTTP方法调用处理程序,传入RequestResponseWriter

处理程序可以是满足Handler您的foo类型所做的接口的任何内容:

type Handler interface {
    ServeHTTP(ResponseWriter, *Request)
}
Run Code Online (Sandbox Code Playgroud)

标准库还包括一些便利,例如HandlerFunc(允许您传递 anyfunc(ResponseWriter, *Request)并将其用作Handler) and ServeMux,它允许您注册许多Handlers 并根据传入的请求路径选择哪个处理哪个请求。