当前位置:网站首页>golang源码分析(12)martini源码分析
golang源码分析(12)martini源码分析
2022-08-02 16:48:00 【用户9710217】
martini是非常优雅的Go Web框架。他基于依赖注入的思想,仿照Sinatra的路由设计,参考Express的中间件设计,而且核心微小,扩展方便,非常值得学习。但是由于本身API设计简洁,使很多细节无法从代码理解。所以,我写一点笔记记录martini
的工作方式。
Martini核心
我们从最简单的官方实例入手:
package main
import "github.com/go-martini/martini"
func main() {
m := martini.Classic()
m.Get("/", func() string {
return "Hello world!"
})
m.Run()
}
martini.Martini
是自带的核心结构,负责完成依赖注入和调用的过程。martini.ClassicMartini
是路由martini.Router
和martini.Martini
的组合,实现路由分发和逻辑调用的过程。m := martini.Classic()
返回的就是martini.ClassicMartini
。具体在martini.go#L104:
func Classic() *ClassicMartini {
r := NewRouter()
m := New()
m.Use(Logger())
m.Use(Recovery())
m.Use(Static("public"))
m.MapTo(r, (*Routes)(nil))
m.Action(r.Handle)
return &ClassicMartini{m, r}
}
里面的m := New()
定义在martini.go#L38:
func New() *Martini {
m := &Martini{Injector: inject.New(), action: func() {}, logger: log.New(os.Stdout, "[martini] ", 0)}
m.Map(m.logger)
m.Map(defaultReturnHandler())
return m
}
依赖注入
上面很明显的看到两个奇特方法:m.Map()
、m.MapTo()
。这里,需要注意martini
的一个最重要原则,注入的任何类型的结构,都是唯一的。即如:
type User struct{
Id int
}
m.Map(&User{Id:1})
m.Map(&User{Id:2})
martini
在寻找&User
类型的时候,只能获取到&User{Id:2}
结构(最后注册的)。Map
的作用,就是向内部注册对应类型的具体对象或者值。类型索引是reflect.Type
。从而我们可以理解到m.New()
的代码中将m.Logger(*log.Logger)
和defaultReturnHandler(martini.ReturnHandler)
(括号中是类型索引)注入到内部。
这里出现了一个问题。接口这种类型,是无法用reflect.Type()
直接获取的(因为传来的都是已经实现接口的具体结构)。解决方法就是m.MapTo()
。
m.MapTo(r, (*Routes)(nil))
即将r(martini.router)
按照martini.Router
接口(注意大小写)类型注入到内部。
(*Routes)(nil)
也是高明的构造。接口的默认值不是nil,无法直接new。但是指针的默认值是nil,可以直接赋值,比如var user *User; user = nil
。因此他注册一个接口指针类型的空指针,用reflect.Type.Elem()
方法就可以获取到指针的内部类型,即接口类型,并以接口类型索引注入到内部。
路由过程
HTTP处理
martini.Martini
实现了http.Handler
方法,实际的HTTP执行过程在代码martini.go#L68:
func (m *Martini) ServeHTTP(res http.ResponseWriter, req *http.Request) {
m.createContext(res, req).run()
}
这里需要我们关注m.createContext
,它返回*martini.context
类型,代码martini.go#L87:
func (m *Martini) createContext(res http.ResponseWriter, req *http.Request) *context {
c := &context{inject.New(), m.handlers, m.action, NewResponseWriter(res), 0}
c.SetParent(m)
c.MapTo(c, (*Context)(nil))
c.MapTo(c.rw, (*http.ResponseWriter)(nil))
c.Map(req)
return c
}
创建*martini.context
类型;然后SetParent
设置寻找注入对象的时候同时从m(*martini.Martini)
中寻找(*martini.context
和*martini.Martini
两个独立的inject
),这样就可以获取m.Map
注入的数据。
这里叉出来说:从代码看出实际上注入的数据有两层,分别在*martini.context
和*martini.Martini
。*martini.context
中的是当前请求可以获取的(每个请求都会m.createContext()
,都是新的对象);martini.Martini
是全局的,任何请求都可以获取到。
回到上一段,c.MapTo
把*martini.context
按martini.Context
接口,将martini.ResponseWriter
按http.ResponseWriter
接口,把req(*http.Request)
注入到当前上下文。
context.run
方法定义在martini.go#L163:
func (c *context) run() {
for c.index <= len(c.handlers) {
_, err := c.Invoke(c.handler())
if err != nil {
panic(err)
}
c.index += 1
if c.Written() {
return
}
}
}
它在循环c.handlers
(来自m.handlers
,createContext代码中)。这里想解释三个细节。
c.Invoke
是inject.Invoke
方法,内部就是获取c.hanlder()
返回的martini.Handler(func)
类型的传入参数reflect.Type.In()
,根据参数个数和类型去内部找对应的结构,然后拼装成[]reflect.Value
给函数的reflect.Value(func).Call()
。
c.handler()
的返回来自两个方面,c.hanlders
和c.action
。c.handlers
来自m.Use()
添加,c.action
来自r.Handle(*martini.router.Handle)
(见上文martini.ClassicMartini.New
中的m.Action(r.Handle)
)。因此,可以发现实际上handlers是有两个列表,一个是c.handlers([]martini.handler)
和r.handlers(martini.routerContext.handlers)
。而且前者先执行。也就是说无论m.Use
写在哪儿,都要比router添加的func先执行。
c.Written
判断请求是否已经发送。他实际上是判断martini.ResponseWriter.status
是否大于0。因此只要发送了response status,handlers过程就会停止。
路由调用
从上面可以知道,路由调用过程有两个方面:一是m.Use()
添加的handlers,二是路由添加比如m.Get("/",handlers...)
中的handlers。m.Use
的handlers调用就是上文的*martini.context.run
方法,不再赘述。路由中的handlers执行是在router.go#L218:
func (r *route) Handle(c Context, res http.ResponseWriter) {
context := &routeContext{c, 0, r.handlers}
c.MapTo(context, (*Context)(nil))
context.run()
}
和router.go#L315:
func (r *routeContext) run() {
for r.index < len(r.handlers) {
handler := r.handlers[r.index]
vals, err := r.Invoke(handler)
if err != nil {
panic(err)
}
r.index += 1
// if the handler returned something, write it to the http response
if len(vals) > 0 {
ev := r.Get(reflect.TypeOf(ReturnHandler(nil)))
handleReturn := ev.Interface().(ReturnHandler)
handleReturn(r, vals)
}
if r.Written() {
return
}
}
}
如果你已经理解上文中说明,这个过程和martini.context.run
是一样的。唯一这里要解释的是martini.ReturnHandler
。它与很上文中的m.Map(defaultReturnHandler())
遥相呼应。
中间件
从上文不难理解,中间件其实就是martini.Handler
被m.Use
添加到m.handlers
中。这里我们来说明官方的一个中间件martini.Logger()
,实现代码在logger.go:
func Logger() Handler {
return func(res http.ResponseWriter, req *http.Request, c Context, log *log.Logger) {
start := time.Now()
log.Printf("Started %s %s", req.Method, req.URL.Path)
rw := res.(ResponseWriter)
c.Next()
log.Printf("Completed %v %s in %v\n", rw.Status(), http.StatusText(rw.Status()), time.Since(start))
}
}
首先看func的传入参数,http.ResponseWriter
和*http.Request
来自:
c := &context{inject.New(), m.handlers, m.action, NewResponseWriter(res), 0}
// ...
c.MapTo(c.rw, (*http.ResponseWriter)(nil))
c.Map(req)
Context
来自:
context := &routeContext{c, 0, r.handlers}
c.MapTo(context, (*Context)(nil))
*log.Logger
来自:
m := &Martini{Injector: inject.New(), action: func() {}, logger: log.New(os.Stdout, "[martini] ", 0)}
m.Map(m.logger)
然后看rw := res.(ResponseWriter)
。实际上c.rw
是NewReponseWriter(res)
返回的martini.ResponseWriter
类型,一次可以在这里直接转换(注意在外部调用,不是martini包中,要import并写res.(martini.ResponseWriter)
)。
最后是c.Next()
方法,源码在martini.go#L154:
func (c *context) Next() {
c.index += 1
c.run()
}
意思就是index自增,指向下一个handler,c.run
走完所有handler,然后继续中间件里的log.Printf...
。
边栏推荐
- NIO Cup 2022 Niu Ke Summer Multi-School Training Camp 5 ABCDFGHK
- 【二】TS基本类型
- Locking and Concurrency Control (2)
- 【无标题】
- Five speakers: seventy genius_platform software platform development 】 【 turn YUY2 RGB24 implementation source code
- npm install 时,卡住不动,五种解决方法
- 一些与开发者体验有关的话题
- 品牌方发行NFT时,应如何考量实用性?
- 在idea中创建web项目_idea部署web项目
- 【电子器件笔记7】MOS管参数和选型
猜你喜欢
随机推荐
Switch 块、Switch 表达式、Switch 模式匹配,越来越好用的 Switch
Kubernetes:(七)优化大法(江湖失传已久的武林秘籍)
Pytest study notes
npm install报gyp info it worked if it ends with ok
nacos简单使用
Informatica旗下PowerCenter的元数据库解析
LeetCode·每日一题·
在idea中创建web项目_idea部署web项目
我的创作纪念日
金仓数据库KingbaseES安全指南--6.11. PAM身份验证
SQL Statement Basics
One article to understand DI dependency injection in php
js通过两种方式进行对商品价格排序
【300+精选大厂面试题持续分享】大数据运维尖刀面试题专栏(十)
Nacos的基本配置
JWT原理详解_电磁感应现象原理
金仓数据库KingbaseES安全指南--6.13. 关于身份验证的常见问题
几种常见的跨域解决方法
尚硅谷尚品项目汇笔记(三)
Kubernetes:(五)Pod进阶(资源限制、健康检查)