我正在学习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)
}
根据我对添加ServeHTTP(w http.ResponseWriter,r *http.Request)作为函数方法的理解,它调用处理程序接口(如果我没说错的话),现在foo也是类型的处理程序。我还理解http.ListenAndServe接受类型为的输入,所以这就是我的变量handler栏发挥作用的地方。当我运行代码并转到浏览器上的localhost:8080时,会出现“一些文本”。
编辑:实现了接口,是不调用的恰当术语。
问题:
这到底是怎么回事?如何访问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)
}
我从来没有见过像上面这样的HandlerFunc类型声明,它的类型名后面有一个函数。我还看到了方法是如何声明的,但不确定上面的代码中发生了什么。
发布于 2018-04-05 22:10:14
这到底是怎么回事?如何访问ServeHTTP函数?
要回答这个问题,我们需要看看http.ListenAndServe
是如何工作的:
func ListenAndServe(addr string, handler Handler) error {
server := &Server{Addr: addr, Handler: handler}
return server.ListenAndServe()
}
在这里,我们使用给定的地址和处理程序创建一个服务器,并调用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)})
}
此方法只是开始侦听给定的地址,并使用新创建的侦听器调用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)
}
}
从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)
...
}
}
在这里,我们最终调用了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)
}
所以它终于出现了。如果我们没有提供任何处理程序,将使用DefaultServeMux,但是因为我们提供了foo处理程序ServeHTTP,所以从foo get的调用中调用。
就是这样。所有这些都可以在server.go中找到
发布于 2018-04-05 21:25:41
Go的HTTP服务器接受一个要侦听的地址和一个处理程序。在内部,它创建一个TCP侦听器来接受给定地址上的连接,并且每当有请求进入时,它:
通过调用http.Request
ResponseWriter
,
http.ResponseWriter
,以便向ServeHTTP
发送处理程序处理程序可以是满足Handler
接口的任何东西,foo
类型就是这样做的:
type Handler interface {
ServeHTTP(ResponseWriter, *Request)
}
标准库还包括一些便利功能,比如HandlerFunc
(它允许您传递任何func(ResponseWriter, *Request)
并将其用作Handler
)和ServeMux
,它允许您注册许多Handler
,并根据传入的请求路径选择处理哪个请求。
发布于 2018-04-05 17:10:06
net/http中的相关类型为
type Handler interface {
ServeHTTP(ResponseWriter, *Request)
}
一种接口类型。实现此接口的任何具体类型都可用于服务HTTP请求。您的处理程序是foo
类型的,并且foo实现了bar
接口。如果内置HTTP服务器必须处理请求,它将调用bar的ServeHTTP方法。
https://stackoverflow.com/questions/49668070
复制相似问题