• 13.3 controller设计
    • controller作用
    • beego的REST设计
    • 应用指南
    • links

    13.3 controller设计

    传统的MVC框架大多数是基于Action设计的后缀式映射,然而,现在Web流行REST风格的架构。尽管使用Filter或者rewrite能够通过URL重写实现REST风格的URL,但是为什么不直接设计一个全新的REST风格的 MVC框架呢?本小节就是基于这种思路来讲述如何从头设计一个基于REST风格的MVC框架中的controller,最大限度地简化Web应用的开发,甚至编写一行代码就可以实现“Hello, world”。

    controller作用

    MVC设计模式是目前Web应用开发中最常见的架构模式,通过分离 Model(模型)、View(视图)和 Controller(控制器),可以更容易实现易于扩展的用户界面(UI)。Model指后台返回的数据;View指需要渲染的页面,通常是模板页面,渲染后的内容通常是HTML;Controller指Web开发人员编写的处理不同URL的控制器,如前面小节讲述的路由就是URL请求转发到控制器的过程,controller在整个的MVC框架中起到了一个核心的作用,负责处理业务逻辑,因此控制器是整个框架中必不可少的一部分,Model和View对于有些业务需求是可以不写的,例如没有数据处理的逻辑处理,没有页面输出的302调整之类的就不需要Model和View,但是controller这一环节是必不可少的。

    beego的REST设计

    前面小节介绍了路由实现了注册struct的功能,而struct中实现了REST方式,因此我们需要设计一个用于逻辑处理controller的基类,这里主要设计了两个类型,一个struct、一个interface

    1. type Controller struct {
    2. Ct *Context
    3. Tpl *template.Template
    4. Data map[interface{}]interface{}
    5. ChildName string
    6. TplNames string
    7. Layout []string
    8. TplExt string
    9. }
    10. type ControllerInterface interface {
    11. Init(ct *Context, cn string) //初始化上下文和子类名称
    12. Prepare() //开始执行之前的一些处理
    13. Get() //method=GET的处理
    14. Post() //method=POST的处理
    15. Delete() //method=DELETE的处理
    16. Put() //method=PUT的处理
    17. Head() //method=HEAD的处理
    18. Patch() //method=PATCH的处理
    19. Options() //method=OPTIONS的处理
    20. Finish() //执行完成之后的处理
    21. Render() error //执行完method对应的方法之后渲染页面
    22. }

    那么前面介绍的路由add函数的时候是定义了ControllerInterface类型,因此,只要我们实现这个接口就可以,所以我们的基类Controller实现如下的方法:

    1. func (c *Controller) Init(ct *Context, cn string) {
    2. c.Data = make(map[interface{}]interface{})
    3. c.Layout = make([]string, 0)
    4. c.TplNames = ""
    5. c.ChildName = cn
    6. c.Ct = ct
    7. c.TplExt = "tpl"
    8. }
    9. func (c *Controller) Prepare() {
    10. }
    11. func (c *Controller) Finish() {
    12. }
    13. func (c *Controller) Get() {
    14. http.Error(c.Ct.ResponseWriter, "Method Not Allowed", 405)
    15. }
    16. func (c *Controller) Post() {
    17. http.Error(c.Ct.ResponseWriter, "Method Not Allowed", 405)
    18. }
    19. func (c *Controller) Delete() {
    20. http.Error(c.Ct.ResponseWriter, "Method Not Allowed", 405)
    21. }
    22. func (c *Controller) Put() {
    23. http.Error(c.Ct.ResponseWriter, "Method Not Allowed", 405)
    24. }
    25. func (c *Controller) Head() {
    26. http.Error(c.Ct.ResponseWriter, "Method Not Allowed", 405)
    27. }
    28. func (c *Controller) Patch() {
    29. http.Error(c.Ct.ResponseWriter, "Method Not Allowed", 405)
    30. }
    31. func (c *Controller) Options() {
    32. http.Error(c.Ct.ResponseWriter, "Method Not Allowed", 405)
    33. }
    34. func (c *Controller) Render() error {
    35. if len(c.Layout) > 0 {
    36. var filenames []string
    37. for _, file := range c.Layout {
    38. filenames = append(filenames, path.Join(ViewsPath, file))
    39. }
    40. t, err := template.ParseFiles(filenames...)
    41. if err != nil {
    42. Trace("template ParseFiles err:", err)
    43. }
    44. err = t.ExecuteTemplate(c.Ct.ResponseWriter, c.TplNames, c.Data)
    45. if err != nil {
    46. Trace("template Execute err:", err)
    47. }
    48. } else {
    49. if c.TplNames == "" {
    50. c.TplNames = c.ChildName + "/" + c.Ct.Request.Method + "." + c.TplExt
    51. }
    52. t, err := template.ParseFiles(path.Join(ViewsPath, c.TplNames))
    53. if err != nil {
    54. Trace("template ParseFiles err:", err)
    55. }
    56. err = t.Execute(c.Ct.ResponseWriter, c.Data)
    57. if err != nil {
    58. Trace("template Execute err:", err)
    59. }
    60. }
    61. return nil
    62. }
    63. func (c *Controller) Redirect(url string, code int) {
    64. c.Ct.Redirect(code, url)
    65. }

    上面的controller基类已经实现了接口定义的函数,通过路由根据url执行相应的controller的原则,会依次执行如下:

    1. Init() 初始化
    2. Prepare() 执行之前的初始化,每个继承的子类可以来实现该函数
    3. method() 根据不同的method执行不同的函数:GETPOSTPUTHEAD等,子类来实现这些函数,如果没实现,那么默认都是403
    4. Render() 可选,根据全局变量AutoRender来判断是否执行
    5. Finish() 执行完之后执行的操作,每个继承的子类可以来实现该函数

    应用指南

    上面beego框架中完成了controller基类的设计,那么我们在我们的应用中可以这样来设计我们的方法:

    1. package controllers
    2. import (
    3. "github.com/astaxie/beego"
    4. )
    5. type MainController struct {
    6. beego.Controller
    7. }
    8. func (this *MainController) Get() {
    9. this.Data["Username"] = "astaxie"
    10. this.Data["Email"] = "astaxie@gmail.com"
    11. this.TplNames = "index.tpl"
    12. }

    上面的方式我们实现了子类MainController,实现了Get方法,那么如果用户通过其他的方式(POST/HEAD等)来访问该资源都将返回405,而如果是Get来访问,因为我们设置了AutoRender=true,那么在执行完Get方法之后会自动执行Render函数,就会显示如下界面:

    controller设计 - 图1

    index.tpl的代码如下所示,我们可以看到数据的设置和显示都是相当的简单方便:

    1. <!DOCTYPE html>
    2. <html>
    3. <head>
    4. <title>beego welcome template</title>
    5. </head>
    6. <body>
    7. <h1>Hello, world!{{.Username}},{{.Email}}</h1>
    8. </body>
    9. </html>
    • 目录
    • 上一章: 自定义路由器设计
    • 下一节: 日志和配置设计