大家好,我是 polarisxu。
该系列第一篇文章发出后,大家褒贬不一,很正常。选什么,不选什么,大家自己评估,没有什么是最好的。我这个系列,更多只是让大家对 Fiber 有些了解,说不定正好合你胃口呢?
前面对 Fiber 有了大概的印象。今天着重较深入探讨 Fiber 相关功能。
fiber.New
01 配置
大部分 Go 框架,获得实例的函数是不支持配置的,比如 Gin、Echo 等。但 Fiber 框架的 New 函数支持传递配置:
// New creates a new Fiber named instance.
// app := fiber.New()
// You can pass optional configuration options by passing a Config struct:
// app := fiber.New(fiber.Config{
// Prefork: true,
// ServerHeader: "Fiber",
// })
func New(config ...Config) *App
一般情况,使用默认配置即可(即不手动传递配置),但有必要了解下,通过配置,我们能干些什么。
比如,我们希望响应头中,Server 用自定义的。
config := fiber.Config{
ServerHeader: "Go Fiber Framework",
}
app := fiber.New(config)
响应头类似这样:
Content-Length: 12
Content-Type: text/plain; charset=utf-8
Date: Mon, 20 Sep 2021 14:58:45 GMT
Server: Go Fiber Framework
实际上,在前文模板引擎使用的 Views 就是一个配置项。
目前配置 29 项之多,有不少是关于 HTTP 的配置。所有的配置和说明可以在文档找到:https://docs.gofiber.io/api/fiber#config。建议扫一遍,有一个印象,方便将来有需求时知道在这里找。
02 路由
标准库 net/http 的路由比较简单,这大概也是有各种路由库(框架)的原因之一。
最简单的路由莫过于直接匹配,如:
// 请求匹配到 /about
app.Get("/about", func(c *fiber.Ctx) error {
return c.SendString("about")
})
而命名路由(也叫参数路由)是一个强大框架必须的,即提供占位符。比如:
app.Get("/hello/:username", func(c *fiber.Ctx) error {
str := fmt.Sprintf("Hello, %s", c.Params("username"))
return c.SendString(str)
})
/hello//hello/polarisxuHello, polarixu
/hello/:username?
app.Get("/hello/:username?", func(c *fiber.Ctx) error {
str := fmt.Sprintf("Hello, %s", c.Params("username"))
return c.SendString(str)
})
+*+*c.Params("+")c.Params("*")
-.
// http://localhost:3000/flights/LAX-SFO
app.Get("/flights/:from-:to", func(c *fiber.Ctx) error {
fmt.Fprintf(c, "%s-%s\n", c.Params("from"), c.Params("to"))
return nil // LAX-SFO
})
:
因为 Fiber 的目标之一是成为 Go 最快、最清晰的 Web 框架,因此对于更复杂的路由,比如正则表达式,Fiber 不会支持。
Fiber 还提供了方法,返回所有注册的路由信息:
var handler = func(c *fiber.Ctx) error { return nil }
func main() {
app := fiber.New()
app.Get("/john/:age", handler)
app.Post("/register", handler)
data, _ := json.MarshalIndent(app.Stack(), "", " ")
fmt.Println(string(data))
app.Listen(":3000")
}
返回结果如下:
[
[
{
"method": "GET",
"path": "/john/:age",
"params": [
"age"
]
}
],
[
{
"method": "HEAD",
"path": "/john/:age",
"params": [
"age"
]
}
],
[
{
"method": "POST",
"path": "/register",
"params": null
}
]
]
可以辅助排查路由问题。
03 Static
上文介绍了服务静态资源的 Static 方法,这里详细解释下。
index.html
app.Static("/static/", "./public")
以上代码用于项目根目录下 public 目录的文件和文件夹。
此外,Static 方法有第三个可选参数,以便对 Static 行为进行微调,这可以通过 fiber.Static 结构体控制。
// Static defines configuration options when defining static assets.
type Static struct {
// When set to true, the server tries minimizing CPU usage by caching compressed files.
// This works differently than the github.com/gofiber/compression middleware.
// Optional. Default value false
Compress bool `json:"compress"`
// When set to true, enables byte range requests.
// Optional. Default value false
ByteRange bool `json:"byte_range"`
// When set to true, enables directory browsing.
// Optional. Default value false.
Browse bool `json:"browse"`
// The name of the index file for serving a directory.
// Optional. Default value "index.html".
Index string `json:"index"`
// Expiration duration for inactive file handlers.
// Use a negative time.Duration to disable it.
//
// Optional. Default value 10 * time.Second.
CacheDuration time.Duration `json:"cache_duration"`
// The value for the Cache-Control HTTP-header
// that is set on the file response. MaxAge is defined in seconds.
//
// Optional. Default value 0.
MaxAge int `json:"max_age"`
// Next defines a function to skip this middleware when returned true.
//
// Optional. Default: nil
Next func(c *Ctx) bool
}
上文说,默认情况下,对目录访问的索引文件是 index.html,通过 Index 可以改变该行为。如果想要启用目录浏览功能,可以设置 Browse 为 true。
04 路由处理器
在前面提到,Fiber 有对应的方法支持所有 HTTP Method。除此之外,还有两个特殊的方法:Add 和 All。
Add 方法是所有 HTTP Method 对应方法的底层实现,比如 Get 方法:
func (app *App) Get(path string, handlers ...Handler) Router {
return app.Add(MethodHead, path, handlers...).Add(MethodGet, path, handlers...)
}
它底层调用了 Add 方法,做了两次绑定,分别是 HEAD 和 GET,也就是说,对于 Get 方法,支持 HTTP GET 和 HEAD。
我之前写过一篇文章:网友很强大,发现了Go并发下载的Bug。Echo 框架,对于 Get 方法,只是 HTTP GET,不支持 HEAD 请求。目前看,Fiber 的做法更合理。如果你真的只需要 GET,可以通过 Add 方法实现。
而 All 方法表示支持任意 HTTP Method。
05 Mount 和 Group
Mount 方法可以将一个 Fiber 实例挂载到另一个实例。
func main() {
micro := fiber.New()
micro.Get("/doe", func(c *fiber.Ctx) error {
return c.SendStatus(fiber.StatusOK)
})
app := fiber.New()
app.Mount("/john", micro) // GET /john/doe -> 200 OK
log.Fatal(app.Listen(":3000"))
}
Group 是路由分组功能,框架基本会支持该特性,对于 API 版本控制很有用。
func main() {
app := fiber.New()
api := app.Group("/api", handler) // /api
v1 := api.Group("/v1", handler) // /api/v1
v1.Get("/list", handler) // /api/v1/list
v1.Get("/user", handler) // /api/v1/user
v2 := api.Group("/v2", handler) // /api/v2
v2.Get("/list", handler) // /api/v2/list
v2.Get("/user", handler) // /api/v2/user
log.Fatal(app.Listen(":3000"))
}
06 fiber.Ctx 的方法
此外,就是 handler 中的参数 fiber.Ctx,这是一个结构体,包含了众多的方法(不少都是方便开发的方法),在使用时查阅 API 文档,或访问 https://docs.gofiber.io/api/ctx 浏览。
这里介绍几个其他框架可能没有的方法。
// BodyParser binds the request body to a struct.
// It supports decoding the following content types based on the Content-Type header:
// application/json, application/xml, application/x-www-form-urlencoded, multipart/form-data
// If none of the content types above are matched, it will return a ErrUnprocessableEntity error
func (c *Ctx) BodyParser(out interface{}) error
该方法将请求绑定到结构体。(响应的也有 QueryParser 方法,主要处理查询字符串到结构体的绑定)
看一个例子:
type Person struct {
Name string `json:"name" xml:"name" form:"name"`
Pass string `json:"pass" xml:"pass" form:"pass"`
}
app.Post("/login", func(ctx *fiber.Ctx) error {
p := new(Person)
if err := ctx.BodyParser(p); err != nil {
return err
}
log.Println(p.Name) // john
log.Println(p.Pass) // doe
return ctx.SendString("Success")
})
// 运行下面的命令进行测试
// curl -X POST -H "Content-Type: application/json" --data "{\"name\":\"john\",\"pass\":\"doe\"}" localhost:3000/login
// curl -X POST -H "Content-Type: application/xml" --data "<login><name>john</name><pass>doe</pass></login>" localhost:3000/login
// curl -X POST -H "Content-Type: application/x-www-form-urlencoded" --data "name=john&pass=doe" localhost:3000/login
// curl -X POST -F name=john -F pass=doe http://localhost:3000/login
// curl -X POST "http://localhost:3000/login?name=john&pass=doe"
关于获取参数,包括路由参数、查询参数、表单参数,Fiber 都非常友好的提供了可选的默认值形式,也就是说,当没有传递对应值时,我们可以给一个默认值,比如:
// 10 是可选的。以下代码表示,当 page 参数没有传递,page=10
page := ctx.Query("page", 10)
默认值模式(可选参数)在 Fiber 中有大量使用,这能极大为使用者带来方便。
ParamsInt
07 小结
通过本文对 Fiber 内置功能的介绍,我的感受是,Fiber 为开发者提供了很多便利。如果你没有用过其他框架,可能没有那么大的感受。后续文章考虑出一个不同框架相关写法的对比。
下篇文章介绍 Fiber 的中间件~