在实际工作中,大家一定会用到go的web框架。那么,你知道各框架是如何处理http请求的吗?今天就主流的web框架gin
、beego
框架以及go标准库net/http
来总结一下http请求的流程。
首先,我们来看下http包是如何处理请求的。通过以下代码我们就能启动一个http服务,并处理请求:
import (
"net/http"
)
func main() {
// 指定路由
http.Handle("/home", &HomeHandler{})
// 启动http服务
http.ListenAndServe(":8000", nil)
}
type HomeHandler struct {}
// 实现ServeHTTP
func (h *HomeHandler) ServeHTTP(response http.ResponseWriter, request *http.Request) {
response.Write([]byte("Hello World"))
}
当我们输入http://localhost:8000/home
的时候,就会执行到HomeHandler
的ServeHTTP
方法,并返回Hello World
。
那这里为什么要给HomeHandler
定义ServeHTTP
方法,或者说为什么会执行到ServeHTTP
方法中呢?
我们顺着http.ListenAndServe
方法的定义:
func ListenAndServe(addr string, handler Handler) error
发现第二个参数是个Handler
类型,而Handler
是一个定义了ServeHTTP
方法的接口类型:
type Handler interface {
ServeHTTP(ResponseWriter, *Request)
}
似乎有了一点点关联,HomeHandler
类型也实现了ServeHTTP
方法。但我们在main函数中调用http.ListenAndServe(":8000", nil)
的时候第二个参数传递的是nil
,那HomeHandler
里的ServeHTTP
方法又是如何被找到的呢?
我们接着再顺着源码一层一层的找下去可以发现,在/src/net/http/server.go
的第1930行有这么一段代码:
serverHandler{c.server}.ServeHTTP(w, w.req)
有个serverHandler
结构体,包装了c.server
。这里的c
是建立的http连接,而c.server
就是在http.ListenAndServe(":8000", nil)
函数中创建的server
对象:
func ListenAndServe(addr string, handler Handler) error {
server := &Server{Addr: addr, Handler: handler}
return server.ListenAndServe()
}
server
中的Handler
就是http.ListenAndServe(":8000", nil)
传递进来的nil
。
好,我们进入 serverHandler{c.server}.ServeHTTP(w, w.req)
函数中再次查看,就可以发现如下代码:
func (sh serverHandler) ServeHTTP(rw ResponseWriter, req *Request) {
handler := sh.srv.Handler
if handler == nil {
handler = DefaultServeMux
}
...
handler.ServeHTTP(rw, req)
}
/src/net/http/server.go
的第2859行到2862行,就是获取到server
中的Handler
,如果是nil
,则使用默认的DefaultServeMux
,然后调用了hander.ServeHTTP
方法。
继续再看DefaultServeMux
中的ServeHTTP
方法,在/src/net/http/server.go
中的第2416行,发现有一行h, _ := mux.Handler(r)
和h.ServeHTTP
方法的调用。这就是通过请求的路径查找到对应的handler
,然后调用该handler
的ServeHTTP
方法。在开始的实例中,就是我们的HomeHandler
的ServeHTTP
方法。
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)
}
也就是说ServeHTTP
方法是net/http
包中规定好了要调用的,所以每一个页面处理函数都必须实现ServeHTTP方法。
gin框架对http的处理流程本质上都是基于go标准包net/http的处理流程的。 下面我们看下gin框架是如何基于net/http实现对一个请求处理的。 首先我们看通过gin框架是如何启动http服务的:
import (
"github.com/gin-gonic/gin"
)
func main() {
// 初始化gin中自定义的Engine结构体对象
engine := gin.New()
// 添加路由
engine.GET("/", HomeHandler)
// 启动http服务
engine.Run(":8000")
}
func HomeHandler(ctx *gin.Context) {
ctx.Writer.Write([]byte("Hi, this is gin Home page"))
}
我们查看engine.Run
函数的源码,发现也是通过net/http
包启动的http
服务。如下:
func (engine *Engine) Run(addr ...string) (err error) {
defer func() { debugPrintError(err) }()
if engine.isUnsafeTrustedProxies() {
debugPrint("[WARNING] You trusted all proxies, this is NOT safe. We recommend you to set a value.\n" +
"Please check https://pkg.go.dev/github.com/gin-gonic/gin#readme-don-t-trust-all-proxies for details.")
}
address := resolveAddress(addr)
debugPrint("Listening and serving HTTP on %s\n", address)
err = http.ListenAndServe(address, engine.Handler())
return
}
函数较短,在第11行,通过http.ListenAndServe(address, engine.Handler())
函数启动的http
服务。和第一节中的通过go
的标准库net/http
启动的服务方式一样,只不过第二个参数不是nil
,而是engine.Handler()
。
我们继续查看engine.Handler()
函数的源码,发现该函数返回的是一个http.Handler
类型。在源代码中,返回的是engine
对象。这里暂且不讨论使用http2
的情况。也就是说engine
实现了http.Handler
接口,即实现了http.Handler
接口中的ServeHTTP
函数。
func (engine *Engine) Handler() http.Handler {
if !engine.UseH2C {
// 这里直接返回了engine对象
return engine
}
h2s := &http2.Server{}
return h2c.NewHandler(engine, h2s)
}
我们再查看Engine
结构体中实现的方法,发现有ServeHTTP
函数的实现,如下:
// ServeHTTP conforms to the http.Handler interface.
func (engine *Engine) ServeHTTP(w http.ResponseWriter, req *http.Request) {
c := engine.pool.Get().(*Context)
c.writermem.reset(w)
c.Request = req
c.reset()
engine.handleHTTPRequest(c)
engine.pool.Put(c)
}
这里我们主要看第8行的engine.handleHTTPRequest(c)
函数,代码如下:
func (engine *Engine) handleHTTPRequest(c *Context) {
httpMethod := c.Request.Method
rPath := c.Request.URL.Path
//省略代码...
// 根据请求的方法httpMethod和请求路径rPath查找对应的路由
t := engine.trees
for i, tl := 0, len(t); i < tl; i++ {
if t[i].method != httpMethod {
continue
}
root := t[i].root
// 在路由树中找到了该请求路径的路由
value := root.getValue(rPath, c.params, c.skippedNodes, unescape)
if value.params != nil {
c.Params = *value.params
}
if value.handlers != nil {
c.handlers = value.handlers
c.fullPath = value.fullPath
c.Next()
c.writermem.WriteHeaderNow()
return
}
// 省略代码...
}
// 省略代码...
// 没有找到路由,则返回404
c.handlers = engine.allNoRoute
serveError(c, http.StatusNotFound, default404Body)
}
主要看第14行的代码部分,根据请求的路径查找路由,找到了对应的路由,从路由中获取该路径对应的处理函数,赋值给该框架自定义的上下文对象c.handlers
,然后执行c.Next()
函数。
c.Next()
函数实际上就是循环c.handlers
,源码如下:
func (c *Context) Next() {
c.index++
for c.index < int8(len(c.handlers)) {
c.handlers[c.index](c)
c.index++
}
}
而c.handlers
是一个HandlersChain
类型�,如下:
type HandlersChain []HandlerFunc
HandlersChain
类型本质上是一个HandlerFunc
数组,而HandlerFunc
类型的定义如下:
type HandlerFunc func(*Context)
这个函数类型是不是就是在注册路由engine.GET("/", HomeHandler)
时HomeHandler
的类型呢?如下是我们注册路由以及定义HomeHandler
的代码:
import (
"github.com/gin-gonic/gin"
)
func main() {
// 初始化gin中自定义的Engine结构体对象
engine := gin.New()
// 添加路由
engine.GET("/", HomeHandler)
// 启动http服务
engine.Run(":8000")
}
func HomeHandler(ctx *gin.Context) {
ctx.Writer.Write([]byte("Hi, this is gin Home page"))
}
这样就形成了一个处理流程的闭环。我们总结下gin框架对http请求的处理流程。
以下是gin框架处理http请求的全景图:
beego框架启动http服务并监听处理http请求本质上也是使用了标准包net/http中的方法。和gin框架不同的是,beego直接使用net/http包中的Server对象进行启动,而并没有使用http.ListenAndServe方法。但本质上是一样的,http.ListenAndServe方法的底层是也调用了net/http包中的Server对象启动的服务。
首先我们看下beego框架启动http服务的过程:
package main
import (
"github.com/beego/beego/v2/server/web"
beecontext "github.com/beego/beego/v2/server/web/context"
)
func main() {
web.Get("/home", HomeHandler)
web.Run(":8000")
}
func HomeHandler(ctx *beecontext.Context){
ctx.Output.Body([]byte("Hi, this is beego home"))
}
在上述代码中,我们注册了一个 /home
路由,然后再8000
端口上启动了http服务。接下来我们看下web.Run(":8000")
的内部实现:
func Run(params ...string) {
if len(params) > 0 && params[0] != "" {
BeeApp.Run(params[0])
}
BeeApp.Run("")
}
在该函数中,调用了BeeApp
的Run
方法。 这里你会发现有两次BeeApp.Run
调用,为什么要调用两次呢?这里其实不是一个bug。我们进BeeApp.Run
函数就可以知道,其实Run
方法运行后就阻塞了,不会进行最后的BeeApp.Run("")
调用,所以不会出现两次调用。如下在第34行时,实际上是通过通道的输出方式进行了阻塞(这里为进行说明,只列出了相关的代码):
func (app *HttpServer) Run(addr string, mws ...MiddleWare) {
// init...
app.initAddr(addr)
app.Handlers.Init()
addr = app.Cfg.Listen.HTTPAddr
var (
err error
l net.Listener
endRunning = make(chan bool, 1)
)
app.Server.Handler = app.Handlers
if app.Cfg.Listen.EnableHTTP {
go func() {
app.Server.Addr = addr
if app.Cfg.Listen.ListenTCP4 {
// 省略...
} else {
if err := app.Server.ListenAndServe(); err != nil {
logs.Critical("ListenAndServe: ", err)
// 100毫秒 让所有的协程运行完成
time.Sleep(100 * time.Microsecond)
endRunning <- true
}
}
}()
}
// 通过通道进行阻塞
<-endRunning
我们再详细看下BeeApp
实例。BeeApp
是*HttpServer
类型的实例,在导入包时,通过init
函数进行的初始化。其定义如下:
var BeeApp *HttpServer
我们看下HttpServer的结构体包含的主要字段如下:
有两个关键的字段,一个是http.Server
类型的Server
,这个就是用来启动并监听服务。看吧,万变不离其宗,最终启动和监听服务还是使用go标准包中的net/http。
另外一个就是ControllerRegister
类型的Handlers
。这个字段就是用来管理路由和http请求的入口。我们看下ControllerRegister
结构体的关键字段:
在ControllerRegister
中关键的字段也有两个,一个是路由表routers
,一个是进行路由匹配的FilterRouter
类型。
我们再来看ControllerRegister
结构体实现的方法中有一个是ServeHTTP
方法,说明是实现了标准表net/http中的http.Handler
接口,源码如下:
func (p *ControllerRegister) ServeHTTP(rw http.ResponseWriter, r *http.Request) {
ctx := p.GetContext()
ctx.Reset(rw, r)
defer p.GiveBackContext(ctx)
var preFilterParams map[string]string
p.chainRoot.filter(ctx, p.getUrlPath(ctx), preFilterParams)
}
其中第8行的 p.chainRoot.filter(ctx, p.getUrlPath(ctx), preFilterParams)
就是路由匹配的过程。实际的路由匹配和执行过程实际上是在ControllerRegister
的serveHttp
方法中,这里注意和http.Handler
接口的ServerHTTP
方法的首字母的大小写的区别。 serveHttp
方法是在初始化chainRoot
对象时指定的过滤函数,在第13行的newFilterRouter
的第二个参数就是具体的路由匹配函数,如下:
func NewControllerRegisterWithCfg(cfg *Config) *ControllerRegister {
res := &ControllerRegister{
routers: make(map[string]*Tree), //路由表,一个方法一棵树
policies: make(map[string]*Tree),
pool: sync.Pool{
New: func() interface{} {
return beecontext.NewContext()
},
},
cfg: cfg,
filterChains: make([]filterChainConfig, 0, 4),
}
res.chainRoot = newFilterRouter("/*", res.serveHttp, WithCaseSensitive(false))
return res
}
最后,我们再看下路由注册的过程。路由注册有三种方式,这里我们只看其中的一种:用可执行函数进行注册,如下:
web.Get("/home", HomeHandler)
func HomeHandler(ctx *beecontext.Context){
ctx.Output.Body([]byte("Hi, this is beego home"))
}
这里HomeHandler
就是一个函数类型。我们随着web.Get
的源码一路找下去,发现最终会返回一个ControllerInfo
路由信息:
func (p *ControllerRegister) createRestfulRouter(f HandleFunc, pattern string) *ControllerInfo {
route := &ControllerInfo{}
route.pattern = pattern
route.routerType = routerTypeRESTFul
route.sessionOn = p.cfg.WebConfig.Session.SessionOn
route.runFunction = f
return route
}
大家看,第6行的f
就是HomeHandler
这个函数,给路由的runFunction
进行了赋值。 在路由匹配阶段,找到了对应的路由信息后,就执行route.runFunction
即可。
好了,beego框架处理http请求的流程基本就是这样,具体的路由实现我们后续再单独起一篇文章介绍。如下是该框架处理http请求的一个全景图:
image.png
通过以上两个流行的开源框架gin和beego以及go标准包net/http处理http请求的分析,可以得知所有的web框架启动http服务和处理http的流程都是基于go标准包net/http执行的。 其本质流程都都是通过net/http
启动服务,然后调用handler
中的ServeHTTP
方法。而框架只要实现了http.Handler接口中的ServeHTTP
方法,并作为http服务的默认入口,就可以在框架中的ServeHTTP
方法中进行路由分发了。如下图:
本文由哈喽比特于2年以前收录,如有侵权请联系我们。
文章来源:https://mp.weixin.qq.com/s/2FVYnKHXFoj18W62pob_jw
京东创始人刘强东和其妻子章泽天最近成为了互联网舆论关注的焦点。有关他们“移民美国”和在美国购买豪宅的传言在互联网上广泛传播。然而,京东官方通过微博发言人发布的消息澄清了这些传言,称这些言论纯属虚假信息和蓄意捏造。
日前,据博主“@超能数码君老周”爆料,国内三大运营商中国移动、中国电信和中国联通预计将集体采购百万台规模的华为Mate60系列手机。
据报道,荷兰半导体设备公司ASML正看到美国对华遏制政策的负面影响。阿斯麦(ASML)CEO彼得·温宁克在一档电视节目中分享了他对中国大陆问题以及该公司面临的出口管制和保护主义的看法。彼得曾在多个场合表达了他对出口管制以及中荷经济关系的担忧。
今年早些时候,抖音悄然上线了一款名为“青桃”的 App,Slogan 为“看见你的热爱”,根据应用介绍可知,“青桃”是一个属于年轻人的兴趣知识视频平台,由抖音官方出品的中长视频关联版本,整体风格有些类似B站。
日前,威马汽车首席数据官梅松林转发了一份“世界各国地区拥车率排行榜”,同时,他发文表示:中国汽车普及率低于非洲国家尼日利亚,每百户家庭仅17户有车。意大利世界排名第一,每十户中九户有车。
近日,一项新的研究发现,维生素 C 和 E 等抗氧化剂会激活一种机制,刺激癌症肿瘤中新血管的生长,帮助它们生长和扩散。
据媒体援引消息人士报道,苹果公司正在测试使用3D打印技术来生产其智能手表的钢质底盘。消息传出后,3D系统一度大涨超10%,不过截至周三收盘,该股涨幅回落至2%以内。
9月2日,坐拥千万粉丝的网红主播“秀才”账号被封禁,在社交媒体平台上引发热议。平台相关负责人表示,“秀才”账号违反平台相关规定,已封禁。据知情人士透露,秀才近期被举报存在违法行为,这可能是他被封禁的部分原因。据悉,“秀才”年龄39岁,是安徽省亳州市蒙城县人,抖音网红,粉丝数量超1200万。他曾被称为“中老年...
9月3日消息,亚马逊的一些股东,包括持有该公司股票的一家养老基金,日前对亚马逊、其创始人贝索斯和其董事会提起诉讼,指控他们在为 Project Kuiper 卫星星座项目购买发射服务时“违反了信义义务”。
据消息,为推广自家应用,苹果现推出了一个名为“Apps by Apple”的网站,展示了苹果为旗下产品(如 iPhone、iPad、Apple Watch、Mac 和 Apple TV)开发的各种应用程序。
特斯拉本周在美国大幅下调Model S和X售价,引发了该公司一些最坚定支持者的不满。知名特斯拉多头、未来基金(Future Fund)管理合伙人加里·布莱克发帖称,降价是一种“短期麻醉剂”,会让潜在客户等待进一步降价。
据外媒9月2日报道,荷兰半导体设备制造商阿斯麦称,尽管荷兰政府颁布的半导体设备出口管制新规9月正式生效,但该公司已获得在2023年底以前向中国运送受限制芯片制造机器的许可。
近日,根据美国证券交易委员会的文件显示,苹果卫星服务提供商 Globalstar 近期向马斯克旗下的 SpaceX 支付 6400 万美元(约 4.65 亿元人民币)。用于在 2023-2025 年期间,发射卫星,进一步扩展苹果 iPhone 系列的 SOS 卫星服务。
据报道,马斯克旗下社交平台𝕏(推特)日前调整了隐私政策,允许 𝕏 使用用户发布的信息来训练其人工智能(AI)模型。新的隐私政策将于 9 月 29 日生效。新政策规定,𝕏可能会使用所收集到的平台信息和公开可用的信息,来帮助训练 𝕏 的机器学习或人工智能模型。
9月2日,荣耀CEO赵明在采访中谈及华为手机回归时表示,替老同事们高兴,觉得手机行业,由于华为的回归,让竞争充满了更多的可能性和更多的魅力,对行业来说也是件好事。
《自然》30日发表的一篇论文报道了一个名为Swift的人工智能(AI)系统,该系统驾驶无人机的能力可在真实世界中一对一冠军赛里战胜人类对手。
近日,非营利组织纽约真菌学会(NYMS)发出警告,表示亚马逊为代表的电商平台上,充斥着各种AI生成的蘑菇觅食科普书籍,其中存在诸多错误。
社交媒体平台𝕏(原推特)新隐私政策提到:“在您同意的情况下,我们可能出于安全、安保和身份识别目的收集和使用您的生物识别信息。”
2023年德国柏林消费电子展上,各大企业都带来了最新的理念和产品,而高端化、本土化的中国产品正在不断吸引欧洲等国际市场的目光。
罗永浩日前在直播中吐槽苹果即将推出的 iPhone 新品,具体内容为:“以我对我‘子公司’的了解,我认为 iPhone 15 跟 iPhone 14 不会有什么区别的,除了序(列)号变了,这个‘不要脸’的东西,这个‘臭厨子’。