golang的http源码跟踪笔记
http
启动httpHandle
使用http包的ListenAndServe方法,需要提供一个Handler对象
func ListenAndServe(addr string, handler Handler) error {
server := &Server{Addr: addr, Handler: handler}
return server.ListenAndServe()
}
func (srv *Server) ListenAndServe() error {
if srv.shuttingDown() {
return ErrServerClosed
}
addr := srv.Addr
if addr == "" {
addr = ":http"
}
ln, err := net.Listen("tcp", addr)
if err != nil {
return err
}
return srv.Serve(ln)
}
Handler对象中提供处理请求的方法,ServerHttp
type Handler interface {
ServeHTTP(ResponseWriter, *Request)
}
Server循环accept请求
在ListenAndServe方法中,使用Handler构建一个Server对象,最终调用其Server方法
func (srv *Server) Serve(l net.Listener) error {
.....
//包装listener
origListener := l
l = &onceCloseListener{Listener: l}
defer l.Close()
//启动失败直接结束
if err := srv.setupHTTP2_Serve(); err != nil {
return err
}
if !srv.trackListener(&l, true) {
return ErrServerClosed
}
defer srv.trackListener(&l, false)
//绑定上下文等信息
baseCtx := context.Background()
if srv.BaseContext != nil {
baseCtx = srv.BaseContext(origListener)
if baseCtx == nil {
panic("BaseContext returned a nil context")
}
}
.....
ctx := context.WithValue(baseCtx, ServerContextKey, srv)
for {
//循环accept请求
rw, err := l.Accept()
if err != nil {
select {
//如果服务器已经关闭,直接结束
case <-srv.getDoneChan():
return ErrServerClosed
default:
}
if ne, ok := err.(net.Error); ok && ne.Temporary() {
//如果accept失败,等待一个时间周期后continue
.....
return err
}
connCtx := ctx
if cc := srv.ConnContext; cc != nil {
connCtx = cc(connCtx, rw)
if connCtx == nil {