前文参见 Martini 的工作方式
在Martini中是这样定义的
<!-- lang: cpp -->
// Handler can be any callable function. Martini attempts to inject services into the handler's argument list.
// Martini will panic if an argument could not be fullfilled via dependency injection.
type Handler interface{}
事实上 Handler 就是一个函数. Handler 贯穿于多个对象中.
Martini 对象设置Handler的方
Martini.ServeHTTP 方法主要执行了 m.createContext(res, req).run()
Martini 对象在收到 http.Request 的时候动态创建 context, 并把所有的 handlers copy 一份给 context.
所有的 handlers 在 context.run 方法中被遍历 Invoke. 当遇到 error 或者有 Written 发生终止遍历.
Router 有自己独立的 handlers.
每一个RESTful 方法都会构建一个 Route, 通过 RESTful 方法设置 Route handlers.
NotFound 设置 notFounds handlers.
Router.Handle 在 context.run 中被最后 Invoke.
Router.Handle 遍历匹配所有的 routes, 如果匹配到一个route, 通过 context.Invoke(route.Handle) 进行调用.
如果没有匹配到 route, 动态构建 routeContext 并把 notFounds 作为其 handlers 进行 routeContext.run.
route.Handle 方法动态构建 routeContext 并把 route.handlers 作为其 handlers 进行 routeContext.run.
routeContext.run 方法遍历并 Ivoke 所有的 routeContext.handlers, 当遇到 error 或者有 Written 发生终止遍历.
ResponseWriter 是在 Martini.createContext 的时候被动态 MapTo Context了.
ResponseWriter.Before 可以添加 beforeFuncs, beforeFuncs 会在任何 Write 操作前以倒序遍历调用一次.
当然 Before 在某个handler中才能被调用到.
前文参见 Martini 的工作方式 中曾经简单提到, Martini 内置Handler Logger和Recovery 是如何配合 Context.Next 进行工作的. 虽然Logger和Recovery的具体功能很简单.
这非常实用. 在使用 Handler 的时候切勿忘记.
通过上述分析可知
可以用通过下列途径使用 Handler
这些 Handlers, notFounds, beforeFuncs 形成了调用队列, Handlers 和 notFounds 有明确的终止队列条件.
开发者建立了 martini-contrib 组织. 这样的管理方式更开放.
事实上这样符合 Martini Injector 的风格, 组件之间的依赖可以通过 Injector 的 Map/Invoke 机制完成.