上善若水

beego框架使用

2019-10-22  本文已影响0人  henryspace

一,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

入口文件

路由配置

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 方法
// CMS API
type CMSController struct {
    beego.Controller
}

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

}

控制器

type xxxController struct {
    beego.Controller
}
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

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)
}
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)

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 模板语法

func (this *MainController) Get() {
        this.Data["Website"] = "beego.me"
        this.Data["Email"] = "astaxie@gmail.com"
        this.TplName = "index.tpl"
}
{{if .IsHome}}
{{else}}
    {{if .IsAbout}}{{end}}
{{end}}
{{range .Pages}}
    {{.Num}} of {{$.Total}}
{{end}}

beego 单元测试

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的时候需要注意的几点是:

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

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

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

上一篇 下一篇

猜你喜欢

热点阅读