Golang中HTTP服务的分析与设计详解(朗格男表价格及图片)不看后悔

随心笔谈2年前发布 admin
199 0 0

文章摘要

这篇文章介绍了如何在Go语言中实现一个高效且安全的服务器框架。文章主要讨论了以下内容: 1. **服务器实现**:通过`func (srv *Server) Serve(l net.Listener)`函数实现服务器逻辑,使用`goroutine`来处理多个连接请求。 2. **资源管理**:文章详细描述了如何在多个`goroutine`中共享服务器对象,避免资源泄漏。包括设置背景上下文、处理基线上下文,以及在`close()`方法中进行资源管理。 3. **重试逻辑**:当连接请求失败时,文章介绍了如何计算重试时间间隔(`tempDelay`),避免因连接过快导致的资源竞争。重试时间最长不超过1秒,初始重试间隔为5毫秒。 4. **钩子函数**:文章提到使用钩子函数`testHookServerServe`来处理特定的服务器初始化或关闭事件。 5. **连接处理**:在处理连接请求时,文章描述了如何在新的 goroutine 中建立和管理连接,确保服务器能够高效地处理大量请求。 总结来说,文章通过 goroutine 和钩子函数的使用,展示了如何实现一个健壮的服务器框架,确保资源的安全共享和连接重试的高效处理。

func (srv *Server) Serve(l net.Listener) error {
if fn :=testHookServerServe; fn !=nil {
fn(srv, l) // call hook with unwrapped 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”)
}
}
var tempDelay time.Duration // how long to sleep on accept failure
// 将整个Server对象设置进ctx中,在多个goroutinue中共享
ctx :=context.WithValue(baseCtx, ServerContextKey, srv)
for {
rw, err :=l.Accept() // 阻塞等待连接
if err !=nil {
select {
case <-srv.getDoneChan():
return ErrServerClosed
default:
}
if ne, ok :=err.(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”, err, tempDelay)
time.Sleep(tempDelay)
continue
}
return err
}
connCtx :=ctx
if cc :=srv.ConnContext; cc !=nil {
connCtx=cc(connCtx, rw)
if connCtx==nil {
panic(“ConnContext returned nil”)
}
}
tempDelay=0
c :=srv.newConn(rw)
c.setState(c.rwc, StateNew, runHooks) // before Serve can return
go c.serve(connCtx) //Serve a new connection 建立新的连接
}
}

© 版权声明

相关文章