Skip to content

Latest commit

 

History

History
217 lines (185 loc) · 5.35 KB

2014-11-04-http_server.md

File metadata and controls

217 lines (185 loc) · 5.35 KB
date layout title categories tags excerpt
2014-11-04 22:47:30 +0000
post
Go语言学习:http编程
编程语言
go
Go语言的标准库对http编程提供了很多的支持,我们只要几行代码,就可以写一个简单的http server,背后却有很多逻辑

示例

Go语言的标准库对http编程提供了很多的支持,我们只要几行代码,就可以写一个简单的http server:

{% highlight go linenos %} package main

import ( "io" "net/http" "log" )

// hello world, the web server func HelloServer(w http.ResponseWriter, req *http.Request) { io.WriteString(w, "hello, world!\n") }

func main() { http.HandleFunc("/hello", HelloServer) err := http.ListenAndServe(":12345", nil) if err != nil { log.Fatal("ListenAndServe: ", err) } }

{% endhighlight %}

在浏览器中访问http://localhost:12345/hello,输出

hello, world!

几个核心的函数和接口

为了更加理解上面示例,有必要来看看背后的实现。先看函数ListenAndServe:

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

很简单,它只是调用Server.ListenAndServe而已。

Server

Server实现了一个服务程序的基本框架:

// A Server defines parameters for running an HTTP server.
// The zero value for Server is a valid configuration.
type Server struct {
	Addr           string        // TCP address to listen on, ":http" if empty
	Handler        Handler       // handler to invoke, http.DefaultServeMux if nil
...
}

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)})
}

func (srv *Server) Serve(l net.Listener) error {
	defer l.Close()
	var tempDelay time.Duration // how long to sleep on accept failure
	for {
		rw, e := l.Accept()
		if e != nil {
			if ne, ok := e.(net.Error); ok && ne.Temporary() {
				if tempDelay == 0 {
					tempDelay = 5 * time.Millisecond
				} else {
					tempDelay *= 2
				}
				if max := 1 * time.Second; tempDelay > max {
					tempDelay = max
				}
				srv.logf("http: Accept error: %v; retrying in %v", e, tempDelay)
				time.Sleep(tempDelay)
				continue
			}
			return e
		}
		tempDelay = 0
		c, err := srv.newConn(rw)
		if err != nil {
			continue
		}
		c.setState(c.rwc, StateNew) // before Serve can return
		go c.serve()
	}
}


// Serve a new connection.
func (c *conn) serve() {
...
		serverHandler{c.server}.ServeHTTP(w, w.req)
...
}

每个连接由一个单独的goroutine进行处理。连接的处理逻辑交给serverHandler处理。

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是结构体ServeMux的实例,它根据请求的路径,将请求转发给相应的Handler。

ServeMux

type ServeMux struct {
	mu    sync.RWMutex
	m     map[string]muxEntry  ///pattern --> muxEntry
	hosts bool // whether any patterns contain hostnames
}

// ServeHTTP dispatches the request to the handler whose
// pattern most closely matches the request URL.
func (mux *ServeMux) ServeHTTP(w ResponseWriter, r *Request) {
	if r.RequestURI == "*" {
		if r.ProtoAtLeast(1, 1) {
			w.Header().Set("Connection", "close")
		}
		w.WriteHeader(StatusBadRequest)
		return
	}
	h, _ := mux.Handler(r)
	h.ServeHTTP(w, r)
}

ServeMux最终调用注册的Handler。

在启动http server之前,我们需要对每个URI调用函数Handle或者HandleFunc注册一个Handler:

// Handle registers the handler for the given pattern
// in the DefaultServeMux.
// The documentation for ServeMux explains how patterns are matched.
func Handle(pattern string, handler Handler) { DefaultServeMux.Handle(pattern, handler) }

// HandleFunc registers the handler function for the given pattern
// in the DefaultServeMux.
// The documentation for ServeMux explains how patterns are matched.
func HandleFunc(pattern string, handler func(ResponseWriter, *Request)) {
	DefaultServeMux.HandleFunc(pattern, handler)
}

注册Handler:

type HandlerFunc func(ResponseWriter, *Request)

// ServeHTTP calls f(w, r).
func (f HandlerFunc) ServeHTTP(w ResponseWriter, r *Request) {
	f(w, r)
}

// HandleFunc registers the handler function for the given pattern.
func (mux *ServeMux) HandleFunc(pattern string, handler func(ResponseWriter, *Request)) {
	mux.Handle(pattern, HandlerFunc(handler))
}

实际上ServeMux.HandleFunc最终也转换成对ServeMux.Handle的调用。

Handler

Handler是逻辑处理接口,它包含一个ServeHTTP方法,由业务实现该方法,并注册给http server。

// ServeHTTP should write reply headers and data to the ResponseWriter
// and then return.  Returning signals that the request is finished
// and that the HTTP server can move on to the next request on
// the connection.
type Handler interface {
	ServeHTTP(ResponseWriter, *Request)
}