beego框架使用

一,beego简介

1,beego是一个go语言框架,可以用来快速开发API、Web、后端服务等各种应用。
2,beego的安装,包地址即源码github地址,go get命令借助代码管理工具如git通过远程拉取或更新代码包及其依赖包,并自动完成编译和安装。
go get github.com/astaxie/beego
go get github.com/beego/bee
3,文件结构及逻辑
beego在执行逻辑上是一个典型的 MVC 架构:

image.png

二,beego流程及使用

通过使用bee工具的命令可以分别生成web和api两种项目结构

bee new myproject
bee api myapi

入口文件

  • 文件名可以随意命名
  • 初始化路由配置,监听http请求

路由配置

  • 固定式路由 参数:路径,controller地址,函数方法名
package routers

import (
    "myapi/controllers"
    "github.com/astaxie/beego"
)

func init() {
    beego.Router("/", &controllers.MainController{})
    beego.Router("/v1/area/getArea", &controllers.AreaController{}, "post:GetArea")
}
  • 正则路由 正则匹配路径及参数
beego.Router(“/api/:id([0-9]+)“, &controllers.RController{})

beego.Router(“/user/:username([\\w]+)“, &controllers.RController{})

  • 自动匹配路由

beego.AutoRouter(&controllers.ObjectController{})

/object/login   调用 ObjectController 中的 Login 方法
/object/logout  调用 ObjectController 中的 Logout 方法
  • 注解式路由 路由里使用namespace引入文件,函数方法的上方加上router注释
    beego.Include(&CMSController{})
// CMS API
type CMSController struct {
    beego.Controller
}

// @router /staticblock [post]
func (this *CMSController) StaticBlock() {

}

控制器

  • 通过type struct结构体引入实现了类似继承的功能
type xxxController struct {
    beego.Controller
}
  • beego.Controller 实现了接口 beego.ControllerInterface 定义了如下函数:
Init(ct *context.Context, childName string, app interface{})

这个函数主要初始化了 Context、相应的 Controller 名称,模板名,初始化模板参数的容器 Data 

Prepare() 这个函数主要是为了用户扩展用的,用户可以重写这个函数实现类似用户验证之类。

Get() 如果用户请求的 HTTP Method 是 GET, 处理 Get 请求。

Post() 如果用户请求的 HTTP Method 是 POST, 处理 Post 请求。

Delete() 如果用户请求的 HTTP Method 是 DELETE, 处理 Delete 请求。

Put() 如果用户请求的 HTTP Method 是 PUT,处理 Put 请求.

Head() 如果用户请求的 HTTP Method 是 HEAD, 处理 Head 请求。

Patch() 如果用户请求的 HTTP Method 是 PATCH, 处理 Patch 请求.

Options() 如果用户请求的HTTP Method是OPTIONS, 处理 Options 请求。

Finish() 这个函数是在执行完相应的 HTTP Method 方法之后执行的,默认是空, 执行例如数据库关闭,清理数据之类的工作。

Render() error 这个函数主要用来实现渲染模板,如果 beego.AutoRender 为 true 的情况下才会执行。

模型orm

  • 默认支持MySQL、PostgreSQL和Sqlite3 三种数据库
  • 数据库连接及增删改查示例
func init() {
    // set default database
    orm.RegisterDataBase("default", "mysql", "username:password@tcp(127.0.0.1:3306)/db_name?charset=utf8", 30)
    // register model
    orm.RegisterModel(new(User))
    // create table
    orm.RunSyncdb("default", false, true)
}

func main() {
    o := orm.NewOrm()
    user := User{Name: "slene"}
    // insert
    id, err := o.Insert(&user)
    fmt.Printf("ID: %d, ERR: %v\n", id, err)
    // update
    user.Name = "astaxie"
    num, err := o.Update(&user)
    fmt.Printf("NUM: %d, ERR: %v\n", num, err)
    // read one
    u := User{Id: user.Id}
    err = o.Read(&u)
    fmt.Printf("ERR: %v\n", err)
    // delete
    num, err = o.Delete(&u)
    fmt.Printf("NUM: %d, ERR: %v\n", num, err)
}
  • 关联查询,关联方式大体有三种,均需在tag注解中说明:
    1, 一对一orm:"rel(one)" 反向:orm:"reverse(one)"
    2, 一对多orm:"rel(fk)" 反向:orm:"reverse(many)"
    4, 多对多orm:"rel(m2m)" 反向:orm:"reverse(many)"
type Post struct {
    Id    int    `orm:"auto"`
    Title string `orm:"size(100)"`
    User  *User  `orm:"rel(fk)"`
}

var posts []*Post
qs := o.QueryTable("post")
num, err := qs.Filter("User__Name", "slene").All(&posts)

  • 原生sql查询
var maps []orm.Params
num, err := o.Raw("SELECT * FROM user").Values(&maps)
for _,term := range maps{
    fmt.Println(term["id"],":",term["name"])
}
  • 事务处理
o.Begin()
...
user := User{Name: "slene"}
id, err := o.Insert(&user)
if err == nil {
    o.Commit()
} else {
    o.Rollback()
}
  • 高级查询操作符
exact / iexact 等于
contains / icontains 包含
gt / gte 大于 / 大于等于
lt / lte 小于 / 小于等于
startswith / istartswith 以…起始
endswith / iendswith 以…结束
in
isnull
后面以 i 开头的表示:大小写不敏感
  • 高级查询操作符示例:
qs.Filter("name__exact", "slene") // WHERE name = 'slene'
qs.Filter("name__contains", "slene") // WHERE name LIKE BINARY '%slene%'
ids:=[]int{17,18,19,20};  qs.Filter("age__in", ids) // WHERE age IN (17, 18, 19, 20)
qs.Filter("profile__age__gt", 17) // WHERE profile.age > 17
qs.Filter("name__startswith", "slene") // WHERE name LIKE BINARY 'slene%'
qs.Filter("profile__id__isnull", true) // WHERE profile_id IS NULL

//select示例
 orm1.QueryTable(TableName("user")).Filter("username__contains", m.Username).Limit(10, 0).OrderBy("-id").All(&users,"id","username");

// User 包装了下面的查询结果
type User struct {
    Id  int
    Username string
    Age  int
}
var users []User

// 获取 QueryBuilder 对象. 需要指定数据库驱动参数。
// 第二个返回值是错误对象,在这里略过
qb, _ := orm.NewQueryBuilder("mysql")

// 构建查询对象
qb.Select("a.id","a.username","b.birthday").
     From(TableName("user") + " as a").
     InnerJoin(TableName("user_profile") + " as b").On("a.id = b.user_id").
     Where("a.username like '"+ m.Username +"%'").
     OrderBy("a.id").Desc().
     Limit(10).Offset(0)

// 导出 SQL 语句
sql := qb.String()

// 执行 SQL 语句
o := orm.NewOrm()
o.Raw(sql, 20).QueryRows(&users)

beego 模板语法

  • go 统一使用了 {{}} 作为左右标签,没有其他的标签符号。如果您想要修改为其它符号,可以参考 模板标签
  • 使用 . 来访问当前位置的上下文
  • 数据输出
func (this *MainController) Get() {
        this.Data["Website"] = "beego.me"
        this.Data["Email"] = "astaxie@gmail.com"
        this.TplName = "index.tpl"
}
  • if … else … end
{{if .IsHome}}
{{else}}
    {{if .IsAbout}}{{end}}
{{end}}
  • range … end
{{range .Pages}}
    {{.Num}} of {{$.Total}}
{{end}}
  • beego 中支持直接载入文件模板
    {{template "path/to/head.html" .}}

beego 单元测试

  • 只有唯一的参数,必须是 t *testing.T 类型
  • 必须以单词 Test 开头,再组合上首字母大写的单词或词组(一般是被测试的方法名称,如 TestValidateClient)
  • 调用 t.Error 或者 t.Fail 方法指明测试失败(这里我使用了 t.Errorf 来提供更多的细节)
    t.Log 可以用来提供一些失败信息以外的调试信息
  • 测试代码文件名必须是 _test 结尾的形式 something_test.go ,例如:addtion_test.go

1,功能测试
go test user_test.go -v -run="TestGetUser"

// TestGetUser is a sample to run an endpoint test
func TestGetUser(t *testing.T) {

    param := map[string]string{
        "username": "a",
    }

    marshaled, _ := json.Marshal(param)
    r, err := http.NewRequest("Post", "/v1/user/GetUser", bytes.NewBuffer(marshaled))
    if err != nil {
        t.Fatalf("should get user success, but fails to send request, error:%s\n", err)
    }
    w := httptest.NewRecorder()
    beego.BeeApp.Handlers.ServeHTTP(w, r)

    beego.Trace("testing", "TestGetUser", "Code[%d]\n%s", w.Code, w.Body.String())

}

2,性能测试
go test user_test.go -v -bench="BenchmarkGetUser"

// BenchmarkGetUser 
func BenchmarkGetUser(b *testing.B) {

    b.ResetTimer()
 
    for i := 0; i < b.N; i++ {
        param := map[string]string{
            "username": "a",
        }

        marshaled, _ := json.Marshal(param)
        r, err := http.NewRequest("Post", "/v1/user/GetUser", bytes.NewBuffer(marshaled))
        if err != nil {
            b.Fatalf("should get user success, but fails to send request, error:%s\n", err)
        }
        w := httptest.NewRecorder()
        beego.BeeApp.Handlers.ServeHTTP(w, r)

        beego.Trace("testing", "TestGetUser", "Code[%d]\n%s", w.Code, w.Body.String())
    }
}

其他

1,import 导入包时,初始化init函数里用到,但是主体各函数里没用到,引入包名左侧要有个下划线别名,否则会报错,例如数据库连接初始化

2, Go语言要求public的变量必须以大写字母开头,private变量则以小写字母开头, 函数名命名也遵循这个规则,否则小写开头的变量和函数名不能被外部访问

3,JSON输出的时候必须注意,只有导出的字段(首字母是大写)才会被输出,如果修改字段名,那么就会发现什么都不会输出,所以必须通过struct tag(结构体注解)定义来实现。针对JSON的输出,我们在定义struct tag的时候需要注意的几点是:

  • 字段的tag是"-",那么这个字段不会输出到JSON
  • tag中带有自定义名称,那么这个自定义名称会出现在JSON的字段名中
  • tag中如果带有"omitempty"选项,那么如果该字段值为空,就不会输出到JSON串中
  • 如果字段类型是bool, string, int, int64等,而tag中带有",string"选项,那么这个字段在输出到JSON的时候会把该字段对应的值转换成JSON字符串

4,Beego框架orm在执行o.Insert(*p)插入数据时候需要传入指针变量作为参数,原因是因为在插入成功后,会返回id给user

5,当字段是指针类型时,如果没有用orm:"-"进行orm忽略,必须要添加标签来进行表关系设置。

6,多个接口返回字段不一样时,如果公用结构体指针查询的值,对输出的字段不好增减控制,建议接口方法内每次对接口输出的字段自定义结构体

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,402评论 6 499
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,377评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,483评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,165评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,176评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,146评论 1 297
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,032评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,896评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,311评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,536评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,696评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,413评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,008评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,659评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,815评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,698评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,592评论 2 353

推荐阅读更多精彩内容

  • beego orm是一个基于Go进行ORM操作的库,它采用了Go style方式对数据库进行操作,实现了struc...
    副班长国伟阅读 2,020评论 2 5
  • 在开始学习beego的过滤器之前,说一个项目开发中遇到的一个bug,这个bug是hybris自己的,导致项目上线后...
    非典型_程序员阅读 5,987评论 0 1
  • 今天主要来简单学习一下beego,一个比较流行的go框架。前段时间学习了go语言的基础,対go算是有了一个简单了解...
    非典型_程序员阅读 33,628评论 1 12
  • Beego官网本身已经整理的非常详细了,但是作为一个学习者,我还是决定自己好好整理一下,这样在后面使用的时候自己对...
    上下求索zsh阅读 713评论 0 0
  • GoLang-Beego.Godep 介绍你是一个从PHP,Python,Ruby等动态语言转移到Go的web应用...
    燕京博士阅读 1,185评论 0 3