思考并回答以下问题:
简单装饰器
我们通过一个简单的例子来看一下装饰器的简单应用,首先编写一个hello函数:1
2
3
4
5
6
7
8
9
10
11package main
import "fmt"
func hello() {
fmt.Println("Hello World!")
}
func main() {
hello()
}
完成上面代码后,执行会输出“Hello World!”。接下来通过以下方式,在打印“Hello World!”前后各加一行日志:1
2
3
4
5
6
7
8
9
10
11
12
13package main
import "fmt"
func hello() {
fmt.Println("before")
fmt.Println("Hello World!")
fmt.Println("after")
}
func main() {
hello()
}
代码执行后输出:1
2
3before
Hello World!
after
当然我们可以选择一个更好的实现方式,即单独编写一个专门用来打印日志的logger函数,示例如下:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20package main
import "fmt"
func logger(f func()) func() {
return func() {
fmt.Println("before")
f()
fmt.Println("after")
}
}
func hello() {
fmt.Println("Hello World!")
}
func main() {
hello := logger(hello)
hello()
}
可以看到logger函数接收并返回了一个函数,且参数和返回值的函数签名同hello一样。然后我们在原来调用hello()的位置进行如下修改:1
2hello := logger(hello)
hello()
这样我们通过logger函数对hello函数的包装,更加优雅的实现了给hello函数增加日志的功能。执行后的打印结果仍为:1
2
3before
Hello World!
after
其实logger函数也就是我们在Python中经常使用的装饰器,因为logger函数不仅可以用于hello,还可以用于其他任何与hello函数有着同样签名的函数。
当然如果想使用Python中装饰器的写法,我们可以这样做:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22package main
import "fmt"
func logger(f func()) func() {
return func() {
fmt.Println("before")
f()
fmt.Println("after")
}
}
// 给 hello 函数打上 logger 装饰器
@logger
func hello() {
fmt.Println("Hello World!")
}
func main() {
// hello 函数调用方式不变
hello()
}
但很遗憾,上面的程序无法通过编译。因为Go语言目前还没有像Python语言一样从语法层面提供对装饰器语法糖的支持。
装饰器实现中间件
尽管Go语言中装饰器的写法不如Python语言精简,但它被广泛运用于Web开发场景的中间件组件中。比如Gin Web框架的如下代码,只要使用过就肯定会觉得熟悉:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17package main
import "github.com/gin-gonic/gin"
func main() {
r := gin.New()
// 使用中间件
r.Use(gin.Logger(), gin.Recovery())
r.GET("/ping", func(c *gin.Context) {
c.JSON(200, gin.H{
"message": "pong",
})
})
_ = r.Run(":8888")
}
如示例中使用gin.Logger()
增加日志,使用gin.Recovery()
来处理panic异常一样,在Gin框架中可以通过r.Use(middlewares...)
的方式给路由增加非常多的中间件,来方便我们拦截路由处理函数,并在其前后分别做一些处理逻辑。
而Gin框架的中间件正是使用装饰模式来实现的。下面我们借用Go语言自带的http库进行一个简单模拟。这是一个简单的WebServer程序,其监听8888端口,当访问/hello路由时会进入handleHello函数逻辑:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34package main
import (
"fmt"
"net/http"
)
func loggerMiddleware(f http.HandlerFunc) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
fmt.Println("before")
f(w, r)
fmt.Println("after")
}
}
func authMiddleware(f http.HandlerFunc) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
if token := r.Header.Get("token"); token != "fake_token" {
_, _ = w.Write([]byte("unauthorized\n"))
return
}
f(w, r)
}
}
func handleHello(w http.ResponseWriter, r *http.Request) {
fmt.Println("handle hello")
_, _ = w.Write([]byte("Hello World!\n"))
}
func main() {
http.HandleFunc("/hello", authMiddleware(loggerMiddleware(handleHello)))
fmt.Println(http.ListenAndServe(":8888", nil))
}
我们分别使用loggerMiddleware、authMiddleware函数对handleHello进行了包装,使其支持打印访问日志和认证校验功能。如果我们还需要加入其他中间件拦截功能,可以通过这种方式进行无限包装。
启动这个Server来验证下装饰器:
对结果进行简单分析可以看到,第一次请求/hello接口时,由于没有携带认证token,收到了unauthorized响应。第二次请求时携带了token,则得到响应“Hello World!”,并且后台程序打印如下日志:1
2
3before
handlehello
after
这说明中间件执行顺序是先由外向内进入,再由内向外返回。而这种一层一层包装处理逻辑的模型有一个非常形象且贴切的名字,洋葱模型。
但用洋葱模型实现的中间件有一个直观的问题。相比于Gin框架的中间件写法,这种一层层包裹函数的写法不如Gin框架提供的r.Use(middlewares...)
写法直观。
Gin框架源码的中间件和handler处理函数实际上被一起聚合到了路由节点的handlers属性中。其中handlers属性是HandlerFunc类型切片。对应到用http标准库实现的WebServer中,就是满足func(ResponseWriter, *Request)
类型的handler切片。
当路由接口被调用时,Gin框架就会像流水线一样依次调用执行handlers切片中的所有函数,再依次返回。这种思想也有一个形象的名字,就叫作流水线(Pipeline)。
接下来我们要做的就是将handleHello和两个中间件loggerMiddleware、authMiddleware聚合到一起,同样形成一个Pipeline。1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44package main
import (
"fmt"
"net/http"
)
type handler func(http.HandlerFunc) http.HandlerFunc
func authMiddleware(f http.HandlerFunc) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
if token := r.Header.Get("token"); token != "fake_token" {
_, _ = w.Write([]byte("unauthorized\n"))
return
}
f(w, r)
}
}
func loggerMiddleware(f http.HandlerFunc) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
fmt.Println("before")
f(w, r)
fmt.Println("after")
}
}
// 聚合handler和middleware
func pipelineHandlers(h http.HandlerFunc, hs ...handler) http.HandlerFunc {
for i := range hs {
h = hs[i](h)
}
return h
}
func handleHello(w http.ResponseWriter, r *http.Request) {
fmt.Println("handle hello")
_, _ = w.Write([]byte("Hello World!\n"))
}
func main() {
http.HandleFunc("/hello", pipelineHandlers(handleHello, loggerMiddleware, authMiddleware))
fmt.Println(http.ListenAndServe(":8888", nil))
}
我们借用pipelineHandlers函数将handler和middleware聚合到一起,实现了让这个简单的WebServer中间件用法跟Gin框架用法相似的效果。
再次启动Server进行验证:
改造成功,跟之前使用洋葱模型写法的结果如出一辙。
总结
简单了解了Go语言中如何实现装饰模式后,我们通过一个WebServer程序中间件,学习了装饰模式在Go语言中的应用。
需要注意的是,尽管Go语言实现的装饰器有类型上的限制,不如Python装饰器那般通用。就像我们最终实现的pipelineHandlers不如Gin框架中间件强大,比如不能延迟调用,通过c.Next()
控制中间件调用流等。但不能因为这样就放弃,因为Go语言装饰器依然有它的用武之地。
Go语言是静态类型语言不像Python那般灵活,所以在实现上要多费一点力气。希望通过这个简单的示例,相信对大家深入学习Gin框架有所帮助。