page contents

Golang beego的路由框架原理介绍

Go 自带的路由器有几个限制: 不支持参数设定,例如 /user/:uid 这种泛类型匹配 (url & 传参对 SEO 不友好, 对用户也不友好) 无法很好的支持 REST 模式 ):_: 我在工作中也只是用到 get p...

attachments-2021-07-qbpVF00k60e66b5d8ae2e.png

Go 自带的路由器有几个限制:

不支持参数设定,例如 /user/:uid 这种泛类型匹配 (url & 传参对 SEO 不友好, 对用户也不友好)

无法很好的支持 REST 模式 ):_: 我在工作中也只是用到 get post 。。。

一般网站的路由规则太多了,编写繁琐 ( 新手的我无法理解大佬的极致精神

让我们跟着大佬的思想设计一个 route register 实现 ,看招~~~

先定义一个储存单元路由的信息

type controllerInfo struct {
regex  *regexp.Regexp  //保存正则路由
params map[int]string  // 路由上的单元正则匹配
controllerType reflect.Type  // 路由对应的控制器
}

定义一个 App 结构体 保存 app 的基本信息

type App struct {
Name string
}

所有路由的集合 方便遍历等等

type ControllerRegistor struct {
routers []*controllerInfo  // 保存所有路由的信息
Application *App  // 应用基本信息
}

公共的 Controller 方法 所有的 控制器 都要继承这个 Controller

type ControllerInterface interface {
}

测试控制器

type UserController struct {
ControllerInterface
}
func (u *UserController) String(){
fmt.Println("i am UserController")
}
func (u *UserController) Get(){
fmt.Println("user get request")
}

注册路由

func (p *ControllerRegistor) Add(pattern string, c ControllerInterface) {
parts := strings.Split(pattern, "/")  // 根据 / 切割 url
j := 0
params := make(map[int]string)  // 保存匹配到的URL参数 :id 之类
for i, part := range parts {
if strings.HasPrefix(part, ":") {  // 匹配到 url 上的参数了
expr := "([^/]+)"  // 匹配所有 的参数值
// 匹配用户自定义的正则
if index := strings.Index(part, "("); index != -1 {  // 匹配到了用户自定义的正则
expr = part[index:]
part = part[:index]
}
params[j] = part  // 保存正则匹配到的键  比如 url 上的 :id 作为键
parts[i] = expr   // 把 url 上的 :id 参数 替换成正则表达式
j++
}
}
pattern = strings.Join(parts, "/")  // 用 / 把切片组合起来
regex, regexErr := regexp.Compile(pattern)  // 校验正则 返回正则实例
if regexErr != nil {
panic(regexErr)  // 简单写法 可以根据自己的需求定制
return
}
// 现在开始创建路由
t := reflect.Indirect(reflect.ValueOf(c)).Type()   // 根据值 找到对应的 类型 也就是 对应的控制器
fmt.Println("url 匹配到的 controller : ", t)
route := &controllerInfo{}
route.regex = regex
route.params = params
route.controllerType = t
p.routers = append(p.routers, route)
}

设置静态路由 转发静态资源

var StaticDir map[string]string
func init(){
StaticDir = make(map[string]string)
}
func (p *ControllerRegistor) SetStaticPath(url string, path string) {
StaticDir[url] = path
}

转发路由到对应的方法上

func (p *ControllerRegistor) ServeHTTP(w http.ResponseWriter, r *http.Request) {
defer func(){
if err := recover(); err != nil {
panic(err)  // 暴力输出 测试
}
}()
var started bool
for prefix, staticDir := range StaticDir {
if strings.HasPrefix(r.URL.Path, prefix) {  // 匹配到静态路由
file := staticDir + r.URL.Path[len(prefix):]  // 重写静态路由
http.ServeFile(w, r, file)    // 调用 go 自带的静态文件处理函数 返回静态资源
started = true
return
}
}
requestPath := r.URL.Path
// 开始比对路由
for _, route := range p.routers {
if !route.regex.MatchString(requestPath) {
continue
}
// 正则匹配路由 并获取 url 上带的参数
matches := route.regex.FindStringSubmatch(requestPath)
if len(matches[0]) != len(requestPath) {
continue
}
params := make(map[string]string)
fmt.Println("start r.URL.RawQuery : ", r.URL.RawQuery)
if len(route.params) > 0 {  // 说明 url 上有参数
values := r.URL.Query()   // 获取url get 传参
fmt.Println("start r.URL.Query : ", values)
for i, match := range matches[1:] {
fmt.Println("matches[",i,"] : ", route.params[i])
values.Add(route.params[i], match)
fmt.Println("params[",i,"] added , r.URL.Query : ", values)
params[route.params[i]] = match
}
r.URL.RawQuery = url.Values(values).Encode() + "&" + r.URL.RawQuery  // 添加参数到 原始url 上
fmt.Println("changed r.URL.RawQuery : ", r.URL.RawQuery)
}
// 利用反射机制 获取 绑定的 控制器
vc := reflect.New(route.controllerType)
method := vc.MethodByName("Miss")
//init := vc.MethodByName("init")
if r.Method == "GET" {
method = vc.MethodByName("Get")
} else if r.Method == "POST" {
method = vc.MethodByName("Post")
} else {  // 其它 RESTFUL 风格就不写了 如果有需要请读者 自己 else if 下去
}
in := make([]reflect.Value, 0)
method.Call(in)
started = true
break
}
if started == false {
http.NotFound(w, r)
}
}

跑起来

func main(){
p := &ControllerRegistor{}
p.SetStaticPath("/img/", "/static/img/")
p.Add("/user/:uid([0-9]+)", &UserController{})
fmt.Println("regex : ",p.routers[0].regex)
fmt.Println("params : ",p.routers[0].params)
fmt.Println("start listen")
err := http.ListenAndServe("127.0.0.1:9999", p)
if err != nil {
fmt.Println("listen failed : ", err)
}
}

更多相关技术内容咨询欢迎前往并持续关注六星社区了解详情。

程序员编程交流QQ群:805358732

如果你想用Python开辟副业赚钱,但不熟悉爬虫与反爬虫技术,没有接单途径,也缺乏兼职经验
关注下方微信公众号:Python编程学习圈,获取价值999元全套Python入门到进阶的学习资料以及教程,还有Python技术交流群一起交流学习哦。

attachments-2022-06-sVBVh87462ad3625bfcab.jpeg

  • 发表于 2021-07-08 11:07
  • 阅读 ( 579 )
  • 分类:Golang

0 条评论

请先 登录 后评论
轩辕小不懂
轩辕小不懂

2403 篇文章

作家榜 »

  1. 轩辕小不懂 2403 文章
  2. 小柒 1474 文章
  3. Pack 1135 文章
  4. Nen 576 文章
  5. 王昭君 209 文章
  6. 文双 71 文章
  7. 小威 64 文章
  8. Cara 36 文章