欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

go標(biāo)準(zhǔn)庫net/http服務(wù)端的實(shí)現(xiàn)示例

 更新時(shí)間:2024年07月22日 08:32:57   作者:StarSky-yuan  
go的http標(biāo)準(zhǔn)庫非常強(qiáng)大,本文主要介紹了go標(biāo)準(zhǔn)庫net/http服務(wù)端,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

1、http簡(jiǎn)單使用

go的http標(biāo)準(zhǔn)庫非常強(qiáng)大,調(diào)用了兩個(gè)函數(shù)就能夠?qū)崿F(xiàn)一個(gè)簡(jiǎn)單的http服務(wù):

func HandleFunc(pattern string, handler func(ResponseWriter, *Request))
func ListenAndServe(addr string, handler Handler) error

handleFunc注冊(cè)一個(gè)路由和相應(yīng)的處理函數(shù),第一個(gè)參數(shù)表示注冊(cè)的路由,第二個(gè)參數(shù)表示注冊(cè)路由對(duì)應(yīng)的處理函數(shù);ListenAndServe用來啟動(dòng)http服務(wù)并監(jiān)聽,第一個(gè)參數(shù)是服務(wù)器地址,第二個(gè)參數(shù)表示使用的處理器。

下面是用這兩個(gè)函數(shù)實(shí)現(xiàn)的簡(jiǎn)單的http服務(wù):注冊(cè)了一個(gè)“/”路由的處理函數(shù),并在8080端口啟動(dòng)http服務(wù),ListenAndServe第二個(gè)參數(shù)為空表示使用標(biāo)準(zhǔn)庫默認(rèn)的處理器,也可使用自定義處理器,傳參即可。處理器的概念在下面標(biāo)準(zhǔn)庫分析中進(jìn)行介紹。

import (
    "net/http"
)


func main() {
    http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
        // TODO
    })


    http.ListenAndServe(":8080", nil)
}

2、http標(biāo)準(zhǔn)庫分析

根據(jù)上面的兩個(gè)函數(shù)來對(duì)http標(biāo)準(zhǔn)庫展開分析

2.1、服務(wù)端數(shù)據(jù)結(jié)構(gòu)

首先介紹下這兩個(gè)函數(shù)涉及到的數(shù)據(jù)類型

(1)服務(wù)器對(duì)象,其中最核心的是Handler成員,表示整個(gè)http服務(wù)的路由器,存儲(chǔ)路由路徑對(duì)應(yīng)到處理函數(shù)的映射,可自定義,例如第1小姐中的案例,沒有自定義路由器對(duì)象,就會(huì)使用標(biāo)準(zhǔn)庫提供的默認(rèn)對(duì)象DefaultServeMux

type Server struct {
	Addr string // 地址  host:port
	Handler Handler // 處理器對(duì)象或路由器
    // ...
}

(2)Handler是一個(gè)接口,提供了ServeHTTP方法,用來將路由映射到相應(yīng)的處理函數(shù)上

type Handler interface {
	ServeHTTP(ResponseWriter, *Request)
}

(3)路由器對(duì)象ServeMux,用來存儲(chǔ)路由到處理函數(shù)的映射關(guān)系,該對(duì)象就是Handler接口的具體實(shí)現(xiàn)。

type serveMux struct {
	mu    sync.RWMutex
	m     map[string]muxEntry
	es    []muxEntry // slice of entries sorted from longest to shortest.
	hosts bool       // whether any patterns contain hostnames
}

(4)muxEntry就是一個(gè)映射關(guān)系單元

type muxEntry struct {
	h       Handler
	pattern string
}

2.2、HandleFunc流程

http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
    // TODO
})

根據(jù)上面的函數(shù)來分析標(biāo)準(zhǔn)庫的執(zhí)行流程,首先看HandleFunc相關(guān)的實(shí)現(xiàn):使用默認(rèn)的DefaultServeMux路由器對(duì)象,調(diào)用ServeMux的HandleFunc,最后路由的注冊(cè)是在mux.handle中實(shí)現(xiàn),其中mux.Handle(pattern, HandlerFunc(handler))中對(duì)處理器做了類型轉(zhuǎn)換,HandlerFunc 類型實(shí)現(xiàn)了ServeHTTP方法,所以被該類型轉(zhuǎn)換后的函數(shù)都是Handler對(duì)象的實(shí)例

var DefaultServeMux = &defaultServeMux

var defaultServeMux ServeMux

type HandlerFunc func(ResponseWriter, *Request)

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

func HandleFunc(pattern string, handler func(ResponseWriter, *Request)) {
    DefaultServeMux.HandleFunc(pattern, handler)
}

func (mux *ServeMux) HandleFunc(pattern string, handler func(ResponseWriter, *Request)) {
	if handler == nil {
		panic("http: nil handler")
	}
	mux.Handle(pattern, HandlerFunc(handler))
}

func (mux *ServeMux) Handle(pattern string, handler Handler) {
	mux.handle(pattern, handler)
}

進(jìn)入到mux.handle中,會(huì)創(chuàng)建一個(gè)路由單元muxEntry對(duì)象,存儲(chǔ)相應(yīng)的路由和處理函數(shù),其中對(duì)于根路徑的存儲(chǔ)需要做出特殊處理,在muxEntry中通過es存儲(chǔ),并按照順序存儲(chǔ)在muxEntry切片中,到此,已經(jīng)完成了路由注冊(cè)

func (mux *ServeMux) handle(pattern string, handler Handler) {
	mux.mu.Lock()
	defer mux.mu.Unlock()

	if pattern == "" {
		panic("http: invalid pattern")
	}
	if handler == nil {
		panic("http: nil handler")
	}
	if _, exist := mux.m[pattern]; exist {
		panic("http: multiple registrations for " + pattern)
	}

	if mux.m == nil {
		mux.m = make(map[string]muxEntry)
	}
	e := muxEntry{h: handler, pattern: pattern}
	mux.m[pattern] = e
	if pattern[len(pattern)-1] == '/' {
		mux.es = appendSorted(mux.es, e)
	}

	if pattern[0] != '/' {
		mux.hosts = true
	}
}

func appendSorted(es []muxEntry, e muxEntry) []muxEntry {
	n := len(es)
	i := sort.Search(n, func(i int) bool {
		return len(es[i].pattern) < len(e.pattern)
	})
	if i == n {
		return append(es, e)
	}

	es = append(es, muxEntry{}) 
	copy(es[i+1:], es[i:])      
	es[i] = e
	return es
}

2.3、ListenAndServe流程

ListenAndServe先初始化一個(gè)Server對(duì)象,并綁定地址和路由器,調(diào)用Server的ListenAndServe方法,其中net.Listen("tcp", addr)用于創(chuàng)建一個(gè)監(jiān)聽套接字并開始監(jiān)聽指定網(wǎng)絡(luò)地址上的連接,返回一個(gè)實(shí)現(xiàn)了Listener接口的對(duì)象。關(guān)鍵是srv.Serve()

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

在srv.Serve(ln)中使用onceCloseListener 對(duì)Listener進(jìn)行封裝,防止被多次關(guān)閉,context.WithValue用來將srv服務(wù)器對(duì)象信息存儲(chǔ)在context中,并使用for循環(huán)輪詢等待連接,l.Accept()會(huì)阻塞等待,直到連接到達(dá),并執(zhí)行conn.serve函數(shù)。

type onceCloseListener struct {
	net.Listener
	once     sync.Once
	closeErr error
}

type contextKey struct {
	name string
}

ServerContextKey = &contextKey{"http-server"}

func (srv *Server) Serve(l net.Listener) error {
	l = &onceCloseListener{Listener: l}
	defer l.Close()
    // ...
	ctx := context.WithValue(baseCtx, ServerContextKey, srv)
	for {
		rw, err := l.Accept()
        // ...
		connCtx := ctx
        // ...
		c := srv.newConn(rw)
        // ...
		go c.serve(connCtx)
	}
}

其中newConn會(huì)將Accept的返回的net.Conn封裝成一個(gè)conn對(duì)象,對(duì)每個(gè)請(qǐng)求都會(huì)創(chuàng)建一個(gè)線程來處理,在conn.serve中會(huì)針對(duì)conn對(duì)象創(chuàng)建讀寫器并將內(nèi)容置入緩沖區(qū),在for中調(diào)用readRequest函數(shù)傳入上下文,在readRequest中讀取請(qǐng)求體req,并返回一個(gè)ResponseWriter的接口對(duì)象,用于向請(qǐng)求方返回響應(yīng),并在調(diào)用serverHandler的ServeHTTP方法

type conn struct {
	server *Server
	rwc net.Conn
    // ...
}

func (c *conn) serve(ctx context.Context) {
	c.remoteAddr = c.rwc.RemoteAddr().String()

	ctx = context.WithValue(ctx, LocalAddrContextKey, c.rwc.LocalAddr())
	ctx, cancelCtx := context.WithCancel(ctx)
	c.cancelCtx = cancelCtx
	defer cancelCtx()

	c.r = &connReader{conn: c}
	c.bufr = newBufioReader(c.r)
	c.bufw = newBufioWriterSize(checkConnErrorWriter{c}, 4<<10)

	for {
		w, _ := c.readRequest(ctx)
		serverHandler{c.server}.ServeHTTP(w, w.req)
		w.finishRequest()
		...
	}
}

serverHandler的ServeHTTP方法用來根據(jù)路由分配handler,如果Server的Handler為空就是用默認(rèn)的DefaultServerMux,對(duì)應(yīng)上了文章一開始調(diào)用ListenAndServe的第二個(gè)參數(shù),如果為空就使用默認(rèn)路由器對(duì)象,最后調(diào)用路由器的ServeHTTP函數(shù)。

type serverHandler struct {
	srv *Server
}

func (sh serverHandler) ServeHTTP(rw ResponseWriter, req *Request) {
	handler := sh.srv.Handler
	if handler == nil {
		handler = DefaultServeMux
	}
	if !sh.srv.DisableGeneralOptionsHandler && req.RequestURI == "*" && req.Method == "OPTIONS" {
		handler = globalOptionsHandler{}
	}

	handler.ServeHTTP(rw, req)
}

接下來流程如下,依次調(diào)用返回命中的handler,如果沒有命中,則采用模糊匹配命中,最后調(diào)用handler的ServeHTTP函數(shù),因?yàn)樽?cè)路由時(shí)候的函數(shù)在注冊(cè)時(shí)候被強(qiáng)轉(zhuǎn)成HandleFunc函數(shù)類型,該類型是實(shí)現(xiàn)ServeHTTP方法的,所以執(zhí)行handler的ServeHTTP方法就是執(zhí)行注冊(cè)路由是對(duì)應(yīng)的處理函數(shù)。

func (mux *ServeMux) ServeHTTP(w ResponseWriter, r *Request) {
    h, _ := mux.Handler(r)
    h.ServeHTTP(w, r)
}


func (mux *ServeMux) Handler(r *Request) (h Handler, pattern string) {
    // ...
    return mux.handler(host, r.URL.Path)
}

func (mux *ServeMux) handler(host, path string) (h Handler, pattern string) {
	mux.mu.RLock()
	defer mux.mu.RUnlock()

    return mux.match(path)
}

func (mux *ServeMux) match(path string) (h Handler, pattern string) {
	v, ok := mux.m[path]
	if ok {
		return v.h, v.pattern
	}

	for _, e := range mux.es {
		if strings.HasPrefix(path, e.pattern) {
			return e.h, e.pattern
		}
	}
	return nil, ""
}

到此這篇關(guān)于go標(biāo)準(zhǔn)庫net/http服務(wù)端的實(shí)現(xiàn)示例的文章就介紹到這了,更多相關(guān)go net/http服務(wù)端內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家! 

相關(guān)文章

  • Centos下搭建golang環(huán)境及vim高亮Go關(guān)鍵字設(shè)置的方法

    Centos下搭建golang環(huán)境及vim高亮Go關(guān)鍵字設(shè)置的方法

    這篇文章先給大家詳細(xì)介紹了在Centos下搭建golang環(huán)境的步驟,大家按照下面的方法就可以自己搭建golang環(huán)境,搭建完成后又給大家介紹了vim高亮Go關(guān)鍵字設(shè)置的方法,文中通過示例代碼介紹的很詳細(xì),有需要的朋友們可以參考借鑒,下面來一起看看吧。
    2016-11-11
  • Golang語言學(xué)習(xí)拿捏Go反射示例教程

    Golang語言學(xué)習(xí)拿捏Go反射示例教程

    這篇文章主要為大家介紹了Golang語言中Go反射示例的教程,教你拿捏Go反射,再也不用被Go反射折磨,有需要的朋友可以共同學(xué)習(xí)參考下
    2021-11-11
  • golang?gorm錯(cuò)誤處理事務(wù)以及日志用法示例

    golang?gorm錯(cuò)誤處理事務(wù)以及日志用法示例

    這篇文章主要為大家介紹了golang?gorm錯(cuò)誤處理事務(wù)以及日志用法示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步早日升職加薪
    2022-04-04
  • golang 實(shí)現(xiàn)對(duì)Map進(jìn)行鍵值自定義排序

    golang 實(shí)現(xiàn)對(duì)Map進(jìn)行鍵值自定義排序

    這篇文章主要介紹了golang 實(shí)現(xiàn)對(duì)Map進(jìn)行鍵值自定義排序,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2021-04-04
  • Go?語言進(jìn)階freecache源碼學(xué)習(xí)教程

    Go?語言進(jìn)階freecache源碼學(xué)習(xí)教程

    這篇文章主要為大家介紹了Go?語言進(jìn)階freecache源碼學(xué)習(xí)教程,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-04-04
  • golang遍歷map的方法小結(jié)

    golang遍歷map的方法小結(jié)

    在Go語言中,使用range關(guān)鍵字可以遍歷map,返回鍵和值,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2025-03-03
  • Go語言實(shí)現(xiàn)釘釘發(fā)送通知

    Go語言實(shí)現(xiàn)釘釘發(fā)送通知

    本文通過代碼給大家介紹了Go語言實(shí)現(xiàn)釘釘發(fā)送通知,代碼簡(jiǎn)單易懂,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2019-11-11
  • 解決Go語言中高頻次和高并發(fā)下隨機(jī)數(shù)重復(fù)的問題

    解決Go語言中高頻次和高并發(fā)下隨機(jī)數(shù)重復(fù)的問題

    在Golang中,獲取隨機(jī)數(shù)的方法一般會(huì)介紹有兩種,一種是基于math/rand的偽隨機(jī),一種是基于crypto/rand的真隨機(jī),math/rand由于其偽隨機(jī)的原理,經(jīng)常會(huì)出現(xiàn)重復(fù)的隨機(jī)數(shù),導(dǎo)致在需要進(jìn)行隨機(jī)的業(yè)務(wù)出現(xiàn)較多的重復(fù)問題,所以本文給大家介紹了較好的解放方案
    2023-12-12
  • golang fmt格式“占位符”的實(shí)例用法詳解

    golang fmt格式“占位符”的實(shí)例用法詳解

    在本篇文章里小編給大家整理的是一篇關(guān)于golang fmt格式“占位符”的實(shí)例用法詳解內(nèi)容,有興趣的朋友們可以學(xué)習(xí)下。
    2021-07-07
  • golang為什么要統(tǒng)一錯(cuò)誤處理

    golang為什么要統(tǒng)一錯(cuò)誤處理

    這篇文章主要介紹了golang為什么要統(tǒng)一錯(cuò)誤處理,統(tǒng)一錯(cuò)誤處理的目的是為了前端開發(fā)接收到后端的statuscode,之后便于前端邏輯上開發(fā)以及開發(fā),下文具體操作過程需要的小伙伴可以參考一下
    2022-04-04

最新評(píng)論