亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Go底層之http標(biāo)準(zhǔn)庫服務(wù)端實(shí)現(xiàn)原理分析

 更新時(shí)間:2025年06月25日 15:24:41   作者:在成都搬磚的鴨鴨  
這篇文章主要介紹了Go底層之http標(biāo)準(zhǔn)庫服務(wù)端實(shí)現(xiàn)原理,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

背景

http協(xié)議的交互框架是C-S架構(gòu),C對應(yīng)客戶端模塊,S對應(yīng)服務(wù)端模塊,接下來我們就基于Go1.23源碼來熟悉http標(biāo)準(zhǔn)庫中服務(wù)端的實(shí)現(xiàn)。

核心數(shù)據(jù)結(jié)構(gòu)

【1】Server對象

位于net/http/server.go文件,其源碼如下:

type Server struct {
	Addr string //服務(wù)器監(jiān)聽規(guī)定TCP地址
	
	Handler Handler //處理http請求的處理器。若為nil,使用http.DefaultServeMux

	DisableGeneralOptionsHandler bool //若為true,將OPTIONS*請求交給Handler處理,否在自動(dòng)響應(yīng)200 OK

	TLSConfig *tls.Config //TLS配置

	ReadTimeout time.Duration //讀取整個(gè)請求(含請求體)的最大超時(shí)時(shí)間。若小于等于0,表示無超時(shí)

	ReadHeaderTimeout time.Duration //單獨(dú)讀取請求頭的超時(shí)時(shí)間。若小于等于0,繼承ReadTimeout的值

	WriteTimeout time.Duration //寫入響應(yīng)的最大超時(shí)時(shí)間。若小于等于0,表示無超時(shí)

	IdleTimeout time.Duration //保持連接空閑的最大時(shí)間(用于keep-Alice)若小于等于0,繼承ReadTimeout的值

	MaxHeaderBytes int //請求頭的最大字節(jié)數(shù)。若為0,使用DefaultMaxHeaderBytes

	TLSNextProto map[string]func(*Server, *tls.Conn, Handler) //用于處理ALPN協(xié)議升級

	ConnState func(net.Conn, ConnState) //客戶端連接狀態(tài)變更時(shí)的回調(diào)函數(shù)(如新建、活躍、空閑、關(guān)閉)

	ErrorLog *log.Logger //自定義錯(cuò)誤日志記錄器。若為nil,則使用標(biāo)準(zhǔn)庫的log包

	BaseContext func(net.Listener) context.Context //為每個(gè)請求生成基礎(chǔ)上下文

	ConnContext func(ctx context.Context, c net.Conn) context.Context //修改新連接的上下文

	inShutdown atomic.Bool //標(biāo)記服務(wù)器是否正在關(guān)閉

	disableKeepAlives atomic.Bool //控制是否禁用Keep-Alive
	nextProtoOnce     sync.Once //確保HTTP/2配置只初始化一次
	nextProtoErr      error     //存儲HTTP/2配置的錯(cuò)誤結(jié)果

	mu         sync.Mutex //保護(hù)listeners和activceConn
	listeners  map[*net.Listener]struct{} //當(dāng)前活躍的監(jiān)聽器集合
	activeConn map[*conn]struct{} //當(dāng)前活躍的連接集合
	onShutdown []func() //服務(wù)器關(guān)閉時(shí)執(zhí)行的鉤子函數(shù)

	listenerGroup sync.WaitGroup //用于等待所有監(jiān)聽器關(guān)閉的同步組
}

【2】Handler對象

位于net/http/server.go文件,其源碼如下:

/*
	Handler接口定義了HTTP請求處理器的標(biāo)準(zhǔn)行為,任何實(shí)現(xiàn)ServerHTTP方法的類型都可以處理HTTP請求。
	Request:包含HTTP請求的所有信息。
	ResponseWriter:用于構(gòu)造HTTP響應(yīng)。
*/
type Handler interface {
	ServeHTTP(ResponseWriter, *Request)
}

【3】ServeMux對象

位于net/http/server.go文件,其源碼如下:

/*
	ServeMux是HTTP請求路由器的核心實(shí)現(xiàn),復(fù)雜將不同URL路徑的請求分發(fā)給對應(yīng)的處理函數(shù)。
	其核心作用是通過內(nèi)部的路由樹(tree)和索引(index)高效匹配請求路徑,同時(shí)通過互斥
	鎖(mu)保證并發(fā)安全,并兼容舊版本路由邏輯(mux121)。
*/
type ServeMux struct {
	mu       sync.RWMutex //保護(hù)路由表的并發(fā)安全(注冊路由時(shí)寫鎖,匹配路由時(shí)讀鎖)
	tree     routingNode //路由前綴樹,存儲URL路徑與處理函數(shù)的映射關(guān)系,支持高效路徑匹配
	index    routingIndex //路由加速索引,優(yōu)化特定場景的查找性能
	patterns []*pattern  //兼容舊版本的路由模式列表
	mux121   serveMux121 //Go1.21及之前舊版本路由實(shí)現(xiàn)
}

服務(wù)端代碼示例

使用http標(biāo)準(zhǔn)庫實(shí)現(xiàn)的一個(gè)簡單http服務(wù)示例如下:

func main() {
	//路由注冊
	http.HandleFunc("POST /xxx", func(writer http.ResponseWriter, request *http.Request) {
		writer.Write([]byte("AAA"))
	})

	//服務(wù)啟用、路由匹配
	http.ListenAndServe(":8000", nil)
}

可以看到一個(gè)http服務(wù)端由兩個(gè)部分組成:路由注冊和服務(wù)啟用,其中服務(wù)啟用內(nèi)部實(shí)現(xiàn)了路由匹配邏輯,接下來再深究內(nèi)部。

路由注冊

路由注冊的入口為http.HandleFunc函數(shù),其源碼如下:

func HandleFunc(pattern string, handler func(ResponseWriter, *Request)) {
	if use121 { 
	    //Go1.21及以前版本路由注冊實(shí)現(xiàn)
		DefaultServeMux.mux121.handleFunc(pattern, handler)
	} else {
	    //新版本路由注冊實(shí)現(xiàn)
		DefaultServeMux.register(pattern, HandlerFunc(handler))
	}
}

舊版本路由注冊實(shí)現(xiàn)源碼如下:

func (mux *serveMux121) handleFunc(pattern string, handler func(ResponseWriter, *Request)) {
	...
	
	//將路徑和對應(yīng)的回調(diào)函數(shù)關(guān)聯(lián)起來,具體關(guān)聯(lián)方法不深究
	mux.handle(pattern, HandlerFunc(handler)) 
}

新版本路由注冊實(shí)現(xiàn)源碼如下:

func (mux *ServeMux) registerErr(patstr string, handler Handler) error {
	...
	
	pat, err := parsePattern(patstr)
	if err != nil {
		return fmt.Errorf("parsing %q: %w", patstr, err)
	}

	...
	
	//關(guān)聯(lián)路徑和對應(yīng)的回調(diào)函數(shù)
	mux.tree.addPattern(pat, handler) 
	mux.index.addPattern(pat)
	mux.patterns = append(mux.patterns, pat)
	return nil
}

舊版本和新版本都會將對應(yīng)的回調(diào)函數(shù)轉(zhuǎn)換為http.HandlerFunc類型,這是因?yàn)閔ttp.HandlerFunc類型實(shí)現(xiàn)了http.Handler接口:

//實(shí)現(xiàn)了Handler接口
type HandlerFunc func(ResponseWriter, *Request)

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

路由匹配

路由匹配的邏輯在服務(wù)啟動(dòng)http.ListenAndServe里,其源碼調(diào)用邏輯鏈路為:

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

func (srv *Server) ListenAndServe() error {
	...
	
	ln, err := net.Listen("tcp", addr)
	if err != nil {
		return err
	}
	return srv.Serve(ln)
}

func (srv *Server) Serve(l net.Listener) error {
	...
	
	for {
		rw, err := l.Accept() //阻塞等待客戶端連接
		
		...

		c := srv.newConn(rw)
		c.setState(c.rwc, StateNew, runHooks) // before Serve can return
		go c.serve(connCtx) //協(xié)程去響應(yīng)客戶端連接
	}
}

func (c *conn) serve(ctx context.Context) {
	...

	for {
		//讀取客戶端請求并構(gòu)造響應(yīng)結(jié)構(gòu)體w
		w, err := c.readRequest(ctx)
	
	    ...
        
        //處理并響應(yīng)請求
		serverHandler{c.server}.ServeHTTP(w, w.req)
		
		...
	}
}

func (sh serverHandler) ServeHTTP(rw ResponseWriter, req *Request) {
	handler := sh.srv.Handler
	if handler == nil { //未定義handler就使用默認(rèn)的http處理器
		handler = DefaultServeMux
	}

	...

	//處理客戶端請求并響應(yīng)
	handler.ServeHTTP(rw, req)
}

//默認(rèn)http處理器實(shí)現(xiàn)的處理客戶端請求并響應(yīng)的方法
func (mux *ServeMux) ServeHTTP(w ResponseWriter, r *Request) {
	...
	
	var h Handler
	if use121 { //舊版本根據(jù)請求進(jìn)行路由匹配
		h, _ = mux.mux121.findHandler(r)
	} else { //新版本根據(jù)請求進(jìn)行路由匹配
		h, r.Pattern, r.pat, r.matches = mux.findHandler(r)
	}
    
    /*
    	這里執(zhí)行的是HandlerFunc類型的ServeHTTP方法,因?yàn)樵谧月酚蓵r(shí)都將回調(diào)函數(shù)轉(zhuǎn)換為了HandlerFunc類型。
    	type HandlerFunc func(ResponseWriter, *Request)

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

總結(jié)

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • Golang中指針的使用詳解

    Golang中指針的使用詳解

    Golang是一門支持指針的編程語言,指針是一種特殊的變量,存儲了其他變量的地址。通過指針,可以在程序中直接訪問和修改變量的值,避免了不必要的內(nèi)存拷貝和傳遞。Golang中的指針具有高效、安全的特點(diǎn),在并發(fā)編程和底層系統(tǒng)開發(fā)中得到廣泛應(yīng)用
    2023-04-04
  • 使用自定義錯(cuò)誤碼攔截grpc內(nèi)部狀態(tài)碼問題

    使用自定義錯(cuò)誤碼攔截grpc內(nèi)部狀態(tài)碼問題

    這篇文章主要介紹了使用自定義錯(cuò)誤碼攔截grpc內(nèi)部狀態(tài)碼問題,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-09-09
  • 深入剖析Go語言中數(shù)組和切片的區(qū)別

    深入剖析Go語言中數(shù)組和切片的區(qū)別

    本文將深入探討 Go 語言數(shù)組和切片的區(qū)別,包括它們的定義、內(nèi)存布局、長度和容量、初始化和操作等方面。從而更好地在實(shí)際開發(fā)中選擇和使用合適的數(shù)據(jù)結(jié)構(gòu),提高代碼的效率和可維護(hù)性,需要的可以參考一下
    2023-05-05
  • Go語言中并發(fā)的工作原理

    Go語言中并發(fā)的工作原理

    本文詳細(xì)講解了Go語言中并發(fā)的工作原理,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-07-07
  • Golang channle管道的基本使用及快速入門

    Golang channle管道的基本使用及快速入門

    管道是Go語言中實(shí)現(xiàn)并發(fā)的一種方式,它可以在多個(gè)goroutine之間進(jìn)行通信和數(shù)據(jù)交換,本文主要介紹了Golang channle管道的基本使用及快速入門,具有一定的參考價(jià)值,感興趣的可以了解一下
    2023-12-12
  • Go語言優(yōu)雅實(shí)現(xiàn)單例模式的多種方式

    Go語言優(yōu)雅實(shí)現(xiàn)單例模式的多種方式

    單例模式(Singleton Pattern)是一種設(shè)計(jì)模式,旨在保證一個(gè)類只有一個(gè)實(shí)例,并且提供全局訪問點(diǎn),單例模式通常用于需要限制某個(gè)對象的實(shí)例數(shù)量為一個(gè)的場景,本文給大家介紹了Go語言實(shí)現(xiàn)單例模式的多種方式,需要的朋友可以參考下
    2025-02-02
  • 使用go自定義prometheus的exporter

    使用go自定義prometheus的exporter

    在prometheus中如果要監(jiān)控服務(wù)器和應(yīng)用的各種指標(biāo),需要用各種各樣的exporter服務(wù),這篇文章主要介紹了使用go自定義prometheus的exporter,需要的朋友可以參考下
    2023-03-03
  • go slice 擴(kuò)容實(shí)現(xiàn)原理源碼解析

    go slice 擴(kuò)容實(shí)現(xiàn)原理源碼解析

    這篇文章主要為大家介紹了go slice 擴(kuò)容實(shí)現(xiàn)原理源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-01-01
  • golang開啟mod后import報(bào)紅的簡單解決方案

    golang開啟mod后import報(bào)紅的簡單解決方案

    這篇文章主要給大家介紹了關(guān)于golang開啟mod后import報(bào)紅的簡單解決方案,文中通過圖文將解決的辦法介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2024-01-01
  • golang?http請求未釋放造成的錯(cuò)誤問題

    golang?http請求未釋放造成的錯(cuò)誤問題

    這篇文章主要介紹了golang?http請求未釋放造成的錯(cuò)誤問題,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-01-01

最新評論