广告位联系
返回顶部
分享到

Beego AutoRouter工作原理解析

Golang 来源:互联网 作者:佚名 发布时间:2022-08-23 19:55:30 人浏览
摘要

一、前言 ???? Beego Web框架应该是国内Go语言社区第一个框架,个人觉得十分适合新手入门Go Web。笔者半年前写过一篇搭建Beego项目并实习简单功能的文章,大家有兴趣可以先看看。 其实

一、前言 ????

Beego Web框架应该是国内Go语言社区第一个框架,个人觉得十分适合新手入门Go Web。笔者半年前写过一篇搭建Beego项目并实习简单功能的文章,大家有兴趣可以先看看。

其实我接触的大部分人都在学校学过Java Web,其实有Java Web的经验,上手Beego也会很舒服。

本文着重讲讲Beego的AutoRouter模块,会结合源码来讲讲,不过由于笔者技术水平有限,如有错误,烦请指出。????

二、从一个例子入手?

Beego的路由设计灵感是sinatra,刚开始并不支持自动路由,项目的每一个路由都需要开发者配置。

???? 不过,在Beego里面注册一个路由是十分简单的,不信你看:

1

2

3

4

import "github.com/beego/beego/v2/server/web"

type ReganYueController struct {

    web.Controller

}

接下来我们可以添加一个方法,也可以重写Get,Post,Delete等方法来响应客户端不同的请求方式。

1

2

3

4

5

6

7

8

9

10

11

import "github.com/beego/beego/v2/server/web"

type ReganYueController struct {

  web.Controller

}

func (u *ReganYueController) HelloWorld()  {

  u.Ctx.WriteString("Welcome, Regan Yue")

}

func main() {

  web.AutoRouter(&ReganYueController{})

  web.Run()

}

该处web.AutoRouter(&ReganYueController{})就是使用的自动路由,如果是以前的话,我们还需要配置路由???? 。例如以下这种形式:

1

beego.Router("/", &IndexController{})

对于下面这段代码,有几点需要注意:

1

2

3

func (u *ReganYueController) HelloWorld()  {

  u.Ctx.WriteString("Welcome, Regan Yue")

}

?这个处理HTTP请求的方法必须是公共方法(首字母要大写),并且不能有参数,不能有返回值,若非如此,可能会发生Panic。

????AutoRouter的解析规则:

影响因素有三:

  • RouterCaseSensitive的值。
  • Controller的名字
  • 方法名字

比如我们上面ReganYueController的名字是ReganYue,而方法名字是HelloWorld,那么就会有以下几种情况出现:

  • 如果RouterCaseSensitive为true,那么AutoRouter就会注册两个路由,其中一个是/ReganYue/HelloWorld/*,另一个是/reganyue/helloworld/*。
  • 如果RouterCaseSensitive为false,那么AutoRouter只会注册一个路由,即/reganyue/helloworld/*。

三、AutoRouter是如何工作的

先看看web.AutoRouter()

1

2

3

4

// AutoRouter see HttpServer.AutoRouter

func AutoRouter(c ControllerInterface) *HttpServer {

  return BeeApp.AutoRouter(c)

}

web.AutoRouter()马上又指向(app *HttpServer) AutoRouter(c ControllerInterface)

1

2

3

4

5

6

7

8

// AutoRouter adds defined controller handler to BeeApp.

// it's same to HttpServer.AutoRouter.

// if beego.AddAuto(&MainContorlller{}) and MainController has methods List and Page,

// visit the url /main/list to exec List function or /main/page to exec Page function.

func (app *HttpServer) AutoRouter(c ControllerInterface) *HttpServer {

  app.Handlers.AddAuto(c)

  return app

}

前面传来的主语BeeApp执行该处程序:

BeeApp是一个应用实例,使用NewHttpSever()创建,继续跟进,发现是根据Bconfig这个配置文件创建的,

1

2

3

4

5

6

7

8

9

10

// NewHttpServerWithCfg will create an sever with specific cfg

func NewHttpServerWithCfg(cfg *Config) *HttpServer {

  cr := NewControllerRegisterWithCfg(cfg)

  app := &HttpServer{

    Handlers: cr,

    Server:   &http.Server{},

    Cfg:      cfg,

  }

  return app

}

上图即配置Bconfig的主要结构。

到此我们对于BeeApp已经有一定了解了,下面我们回过头来看看app.Handlers.AddAuto(c)。

先看看这个c是什么,它的类型是ControllerInterface,我们现在进去看看。

这个c是用来统一所有controller handler的接口。

根据上图我们可以知道,这个app.Handles就是ControllerRegister,再来看看ControllerRegister的AddAuto方法:

1

2

3

func (p *ControllerRegister) AddAuto(c ControllerInterface) {

  p.AddAutoPrefix("/", c)

}

AddAuto又指向AddAutoPrefix,这个AddAutoPrefix有什么用,我们先给出一个例子,然后再来看源码。

1

beego.AddAutoPrefix("/admin",&MainContorlller{})

如果MainContorlller有两个方法List、Page。那么我们可以访问/admin/main/list来执行List函数,访问/admin/main/page来执行Page函数

来看看ControllerRegister的AddAutoPrefix方法:

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

func (p *ControllerRegister) AddAutoPrefix(prefix string, c ControllerInterface) {

    //对传入的Controller做反射

  reflectVal := reflect.ValueOf(c)

    //获取传入的Controller的类型

  rt := reflectVal.Type()

    //因为c是指针,所以要用Indirect方法获取指针指向的变量类型

  ct := reflect.Indirect(reflectVal).Type()

    //使用Beego注册controller的名称后面有Controller,这里把它去掉得到controllerName。

  controllerName := strings.TrimSuffix(ct.Name(), "Controller")

    //

  for i := 0; i < rt.NumMethod(); i++ {

    if !utils.InSlice(rt.Method(i).Name, exceptMethod) {

      route := &ControllerInfo{}

      route.routerType = routerTypeBeego

      route.methods = map[string]string{"*": rt.Method(i).Name}

      route.controllerType = ct

      pattern := path.Join(prefix, strings.ToLower(controllerName), strings.ToLower(rt.Method(i).Name), "*")

      patternInit := path.Join(prefix, controllerName, rt.Method(i).Name, "*")

      patternFix := path.Join(prefix, strings.ToLower(controllerName), strings.ToLower(rt.Method(i).Name))

      patternFixInit := path.Join(prefix, controllerName, rt.Method(i).Name)

      route.pattern = pattern

      for m := range HTTPMETHOD {

        p.addToRouter(m, pattern, route)

        p.addToRouter(m, patternInit, route)

        p.addToRouter(m, patternFix, route)

        p.addToRouter(m, patternFixInit, route)

      }

    }

  }

}

reflectVal.Type()直接的获取传入的Controller的类型,而reflect.Indirect(reflectVal).Type(),interface其实就是两个指针,一个指向类型信息,一个指向实际的对象,用Indirect方法获取指针指向的实际变量的类型。

在runtime/runtime2.go可以了解interface其实就是两个指针:

1

2

3

4

5

6

7

8

9

10

11

type iface struct {

        tab  *itab          //类型信息

        data unsafe.Pointer //实际对象指针

}

type itab struct {

        inter *interfacetype //接口类型

        _type *_type         //实际对象类型

        hash  uint32

        _     [4]byte

        fun   [1]uintptr     //实际对象方法地址

}

接下来是for i := 0; i < rt.NumMethod(); i++,我们来看看这个NumMethod(),可以看到这个方法获得interface类型的方法数量。

utils.InSlice()方法正如其名:

1

2

3

4

5

6

7

8

func InSlice(v string, sl []string) bool {

  for _, vv := range sl {

    if vv == v {

      return true

    }

  }

  return false

}

该方法是用来判断字符串v是不是在字符串切片sl里面。

此处判断方法名是不是在exceptMethod里面。

下面是exceptMethod的内容:

1

2

3

4

5

6

7

exceptMethod = []string{"Init", "Prepare", "Finish", "Render", "RenderString",

    "RenderBytes", "Redirect", "Abort", "StopRun", "UrlFor", "ServeJSON", "ServeJSONP",

    "ServeYAML", "ServeXML", "Input", "ParseForm", "GetString", "GetStrings", "GetInt", "GetBool",

    "GetFloat", "GetFile", "SaveToFile", "StartSession", "SetSession", "GetSession",

    "DelSession", "SessionRegenerateID", "DestroySession", "IsAjax", "GetSecureCookie",

    "SetSecureCookie", "XsrfToken", "CheckXsrfCookie", "XsrfFormHtml",

    "GetControllerAndAction", "ServeFormatted"}

接下来创建了一个结构体,记录了controller的信息,下面几行代码就生成了每个方法对应的controller信息。

controller的pattern这里生成了4个模式:

  • prefix/全小写的controllerName/全小写的方法名/*
  • prefix/controllerName/方法名/*
  • prefix/全小写的controllerName/全小写的方法名
  • prefix/controllerName/方法名

然后对每一种HTTP方法:

都使用addToRouter方法用四种模式执行一遍。

下面看看addToRouter。

1

2

3

4

5

6

7

8

9

10

11

12

func (p *ControllerRegister) addToRouter(method, pattern string, r *ControllerInfo) {

  if !p.cfg.RouterCaseSensitive {

    pattern = strings.ToLower(pattern)

  }

  if t, ok := p.routers[method]; ok {

    t.AddRouter(pattern, r)

  } else {

    t := NewTree()

    t.AddRouter(pattern, r)

    p.routers[method] = t

  }

}

  • 如果RouterCaseSensitive为true,那么AutoRouter就会注册两个路由,其中一个是/ReganYue/HelloWorld/*,另一个是/reganyue/helloworld/*。
  • 如果RouterCaseSensitive为false,那么AutoRouter只会注册一个路由,即/reganyue/helloworld/*。

然后将method传给ControllerRegister,看是不是注册成功。

成功就执行:t.AddRouter(pattern, r)添加路由。

否则就执行:

1

2

3

t := NewTree()

t.AddRouter(pattern, r)

p.routers[method] = t

那就到此为止吧,

再爱就不礼貌了...


版权声明 : 本文内容来源于互联网或用户自行发布贡献,该文观点仅代表原作者本人。本站仅提供信息存储空间服务和不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权, 违法违规的内容, 请发送邮件至2530232025#qq.cn(#换@)举报,一经查实,本站将立刻删除。

您可能感兴趣的文章 :

原文链接 : https://juejin.cn/post/7133871580430401544
    Tag :
相关文章
  • 基于GORM实现CreateOrUpdate的方法
    CreateOrUpdate 是业务开发中很常见的场景,我们支持用户对某个业务实体进行创建/配置。希望实现的 repository 接口要达到以下两个要求: 如果
  • Golang中的内存逃逸的介绍
    什么是内存逃逸分析 内存逃逸分析是go的编译器在编译期间,根据变量的类型和作用域,确定变量是堆上还是栈上 简单说就是编译器在编译
  • Golang自旋锁的介绍
    自旋锁 获取锁的线程一直处于活跃状态,但是并没有执行任何有效的任务,使用这种锁会造成busy-waiting。 它是为实现保护共享资源而提出的
  • Go语言读写锁RWMutex的源码

    Go语言读写锁RWMutex的源码
    在前面两篇文章中初见 Go Mutex、Go Mutex 源码详解,我们学习了Go语言中的Mutex,它是一把互斥锁,每次只允许一个goroutine进入临界区,可以保
  • Go项目实现优雅关机与平滑重启功能
    什么是优雅关机? 优雅关机就是服务端关机命令发出后不是立即关机,而是等待当前还在处理的请求全部处理完毕后再退出程序,是一种对
  • Go语言操作Excel利器之excelize类库的介绍
    在开发中一些需求需要通过程序操作excel文档,例如导出excel、导入excel、向excel文档中插入图片、表格和图表等信息,使用Excelize就可以方便
  • 利用Go语言快速实现一个极简任务调度系统

    利用Go语言快速实现一个极简任务调度系统
    任务调度(Task Scheduling)是很多软件系统中的重要组成部分,字面上的意思是按照一定要求分配运行一些通常时间较长的脚本或程序。在爬
  • GoLang中的iface 和 eface 的区别介绍

    GoLang中的iface 和 eface 的区别介绍
    GoLang之iface 和 eface 的区别是什么? iface和eface都是 Go 中描述接口的底层结构体,区别在于iface描述的接口包含方法,而eface则是不包含任何方
  • Golang接口使用的教程
    go语言并没有面向对象的相关概念,go语言提到的接口和java、c++等语言提到的接口不同,它不会显示的说明实现了接口,没有继承、子类、
  • go colly 爬虫实现示例介绍
    贡献某CC,go源码爬虫一个,基于colly,效果是根据输入的浏览器cookie及excel必要行列号,从excel中读取公司名称,查询公司法人及电话号码。
  • 本站所有内容来源于互联网或用户自行发布,本站仅提供信息存储空间服务,不拥有版权,不承担法律责任。如有侵犯您的权益,请您联系站长处理!
  • Copyright © 2017-2022 F11.CN All Rights Reserved. F11站长开发者网 版权所有 | 苏ICP备2022031554号-1 | 51LA统计