Go DB5. 使用Beego orm库进行ORM开发

beego orm是一个基于Go进行ORM操作的库,它采用了Go style方式对数据库进行操作,实现了struct到数据表记录的映射。

beego orm是一个十分轻量级的Go ORM框架,开发这个库的本意降低复杂的ORM学习曲线,尽可能在ORM的运行效率和功能之间寻求一个平衡,beego orm是目前开源的Go ORM框架中实现比较完整的一个库,而且运行效率相当不错,功能也基本能满足需求。

支持的数据库驱动

beego orm是支持database/sql标准接口的ORM库,所以理论上来说,只要数据库驱动支持database/sql接口就可以无缝的接入beego orm。目前测试过的驱动包括下面几个:

Mysql: github.com/go-mysql-driver/mysql

PostgreSQL: github.com/lib/pq

SQLite: github.com/mattn/go-sqlite3

Mysql: github.com/ziutek/mymysql/godrv

安装

beego orm支持go get方式安装,是完全按照Go Style的方式来实现的。

# go get github.com/astaxie/beego

初始化

首先你需要import相应的数据库驱动包、database/sql标准接口包以及beego orm包,如下所示:

import (

    "database/sql"

    "fmt"

    "github.com/astaxie/beego/orm"

    _ "github.com/go-sql-driver/mysql"

)

func init() {

    // 注册驱动

    orm.RegisterDriver("mysql", orm.DRMySQL)

    // 设置默认数据库

    orm.RegisterDataBase("default", "mysql", "root:1234@/test?charset=utf8", 30)

    // 注册定义的model

    orm.RegisterModel(new(User))

    // 创建table

    orm.RunSyncdb("default", false, true)

}

PostgreSQL配置

// 导入驱动

// _ "github.com/lib/pq"

// 注册驱动

orm.RegisterDriver("postgre", orm.DRPostgres)

// 设置默认数据库

orm.RegisterDataBase("default", "postgres", "user=root password=1234 dbname=test host=localhost port=5432 sslmode=disable")

MySQL配置

// 导入驱动

// _ "github.com/go-sql-driver/mysql"

// 注册驱动

orm.RegisterDriver("mysql", orm.DRMySQL)

// 设置默认数据库

orm.RegisterDataBase("default", "mysql", "root:1234@/test?charset=utf8")

Sqlite配置

// 导入驱动

// _ "github.com/mattn/go-sqlite3"

// 注册驱动

orm.RegisterDriver("sqlite", orm.DRSqlite)

// 设置默认数据库

orm.RegisterDataBase("default", "sqlite3", "./datas/test.db")

数据库连接

导入必须的package之后,我们需要打开到数据库的链接,然后创建一个beego orm对象(以MySQL为例),如下所示:

func main() {

    o := orm.NewOrm()

}

示例代码

package main

import (

    "fmt"

    "github.com/astaxie/beego/orm"

    _ "github.com/go-sql-driver/mysql"

)

// Model Struct

type User struct {

    Id  int

    Name string `orm:"size(100)"`

}

func init() {

    // 注册驱动

    orm.RegisterDriver("mysql", orm.DRMySQL)

    // 设置默认数据库

    orm.RegisterDataBase("default", "mysql", "root:1234@/test?charset=utf8", 30)

    // 注册定义的model

    orm.RegisterModel(new(User))

    // 创建table

    orm.RunSyncdb("default", false, true)

}

func main() {

    o := orm.NewOrm()

    user := User{Name: "lai"}

    // 插入表

    id, err := o.Insert(&user)

    fmt.Printf("ID: %d, ERR:%v\n", id, err)

    // 更新表

    user.Name = "laiupdate"

    num, err := o.Update(&user)

    fmt.Printf("NUM:%d, ERR:%v\n", num, err)

    // 读取ONE

    u := User{Id: user.Id}

    err = o.Read(&u)

    fmt.Printf("ERR:%v\n", err)

    // 删除表

    num, err = o.Delete(&u)

    fmt.Printf("NUM:%d, ERR:%v\n", num, err)

}

数据库表结构对应的struct

type Userinfo struct {

    Uid    int `PK` //如果表的主键不是id,那么需要加上pk注释,显式的说这个字段是主键

    Username    string

    Departname  string

    Created    time.Time

}

type User struct {

    Uid          int `PK` //如果表的主键不是id,那么需要加上pk注释,显式的说这个字段是主键

    Name        string

    Profile    *Profile  `orm:"rel(one)"` // OneToOne relation

    Post        []*Post `orm:"reverse(many)"` // 设置一对多的反向关系

}

type Profile struct {

    Id          int

    Age        int16

    User        *User  `orm:"reverse(one)"` // 设置一对一反向关系(可选)

}

type Post struct {

    Id    int

    Title string

    User  *User  `orm:"rel(fk)"`    //设置一对多关系

    Tags  []*Tag `orm:"rel(m2m)"`

}

type Tag struct {

    Id    int

    Name  string

    Posts []*Post `orm:"reverse(many)"`

}

func init() {

    // 需要在init中注册定义的model

    orm.RegisterModel(new(Userinfo),new(User), new(Profile), new(Tag))

}

注意:beego orm针对驼峰命名会自动帮你转化成下划线字段,例如你定义了Struct名字为UserInfo,那么转化成底层实现的时候是user_info,字段命名也遵循该规则。

其他配置

SetMaxIdleConns

根据数据库的别名,设置数据库的最大空闲连接

orm.SetMaxIdleConns("default", 30)

SetMaxOpenConns

根据数据库的别名,设置数据库的最大数据库连接 (go >= 1.2)

orm.SetMaxOpenConns("default", 30)

Debug模式

目前beego orm支持打印调试,你可以通过如下的代码实现调试

orm.Debug = true

插入数据

下面的代码演示了如何插入一条记录,可以看到我们操作的是struct对象,而不是原生的sql语句,最后通过调用Insert接口将数据保存到数据库。

o := orm.NewOrm()

var user User

user.Name = "zxxx"

user.Departname = "zxxx"

id, err := o.Insert(&user)

if err == nil {

    fmt.Println(id)

}

我们看到插入之后user.Uid就是插入成功之后的自增ID。

同时插入多个对象:InsertMulti

类似sql语句

insert into table (name, age) values("slene", 28),("astaxie", 30),("unknown", 20)

第一个参数 bulk 为并列插入的数量,第二个为对象的slice

返回值为成功插入的数量

users := []User{

    {Name: "slene"},

    {Name: "astaxie"},

    {Name: "unknown"},

    ...

}

successNums, err := o.InsertMulti(100, users)

bulk 为 1 时,将会顺序插入 slice 中的数据。

更新数据

继续上面的例子来演示更新操作,现在user的主键已经有值了,此时调用Insert接口,beego orm内部会自动调用update以进行数据的更新而非插入操作。

o := orm.NewOrm()

user := User{Uid: 1}

if o.Read(&user) == nil {

    user.Name = "MyName"

    if num, err := o.Update(&user); err == nil {

        fmt.Println(num)

    }

}

Update 默认更新所有的字段,可以更新指定的字段:

// 只更新 Name

o.Update(&user, "Name")

// 指定多个字段

// o.Update(&user, "Field1", "Field2", ...)

//Where:用来设置条件,支持多个参数,第一个参数如果为整数,相当于调用了Where("主键=?",值)。

查询数据

beego orm的查询接口比较灵活,具体使用请看下面的例子

例子1:根据主键获取数据

o := orm.NewOrm()

var user User

user := User{Id: 1}

err = o.Read(&user)

if err == orm.ErrNoRows {

    fmt.Println("查询不到")

} else if err == orm.ErrMissPK {

    fmt.Println("找不到主键")

} else {

    fmt.Println(user.Id, user.Name)

}

例子2:指定WHERE查询条件

o := orm.NewOrm()

var user User

qs := o.QueryTable(user) // 返回 QuerySeter

qs.Filter("id", 1) // WHERE id = 1

qs.Filter("profile__age", 18) // WHERE profile.age = 18

例子3:WHERE IN查询条件

qs.Filter("profile__age__in", 18, 20)

// WHERE profile.age IN (18, 20)

例子4:更加复杂的条件

qs.Filter("profile__age__in", 18, 20).Exclude("profile__lt", 1000)

// WHERE profile.age IN (18, 20) AND NOT profile_id < 1000

可以通过如下接口获取多条数据,请看示例

例子1,根据条件age>17,获取20位置开始的10条数据的数据

var allusers []User

qs.Filter("profile__age__gt", 17)

// WHERE profile.age > 17

例子2,limit默认从10开始,获取10条数据

qs.Limit(10, 20)

// LIMIT 10 OFFSET 20 注意跟SQL反过来的

删除数据

beedb提供了丰富的删除数据接口,请看下面的例子

例子1,删除单条数据

o := orm.NewOrm()

if num, err := o.Delete(&User{Id: 1}); err == nil {

fmt.Println(num)

}

Delete 操作会对反向关系进行操作,此例中 Post 拥有一个到 User 的外键。删除 User 的时候,如果 on_delete 设置为默认的级联操作,将删除对应的 Post。

关联查询

基于struct的关联查询:

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)

Group By 和 Having

GroupBy:用来指定进行groupby的字段

Having:用来指定having执行的时候的条件

OrderBy

qs.OrderBy("id", "-profile__age")

// ORDER BY id ASC, profile.age DESC

qs.OrderBy("-profile__age", "profile")

// ORDER BY profile.age DESC, profile_id ASC

使用原生SQL

1、简单示例:

o := orm.NewOrm()

var r orm.RawSeter

r = o.Raw("UPDATE user SET name = ? WHERE name = ?", "testing", "slene")

2、复杂原生SQL:

func (m *User) Query(name string) user []User {

    var o orm.Ormer

    var rs orm.RawSeter

    o = orm.NewOrm()

    rs = o.Raw("SELECT * FROM user "+

                "WHERE name=? AND uid>10 "+

                "ORDER BY uid DESC "+

                "LIMIT 100", name)

    //var user []User

    num, err := rs.QueryRows(&user)

    if err != nil {

        fmt.Println(err)

    } else {

        fmt.Println(num)

        //return user

    }

    return

}

更多说明,请到:beego.me

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

推荐阅读更多精彩内容