资料参考
go-kratos v2 + gorm 实现增删改查demo
https://www.cnblogs.com/zhouqi666/p/15160796.html
Go orm框架gorm学习
https://www.cnblogs.com/rickiyang/p/11074162.html
GORM官网
https://gorm.io/zh_CN/docs/query.html
gorm 中文官网 删除记录
https://learnku.com/docs/gorm/v2/delete/9735
gorm 介绍
原生的Go连接MYSQL的方法,使用Go自带的"database/sql"数据库连接api,"github.com/go-sql-driver/mysql"MYSQL驱动,通过比较原生的写法去写sql和处理事务。目前开源界也有很多封装好的orm操作框架,帮我们简省一些重复的操作,提高代码可读性。gorm就是这样的一款作品,我们来学习一下gorm的操作流程。
安装 gorm
go get -u github.com/jinzhu/gorm
go get -u gorm.io/gorm
安装mysql驱动
go get -u gorm.io/driver/mysql
数据库连接
要连接到数据库首先要导入驱动程序。例如
import _ "github.com/go-sql-driver/mysql"
为了方便记住导入路径,GORM包装了一些驱动。
import _ "github.com/jinzhu/gorm/dialects/mysql"
// import _ "github.com/jinzhu/gorm/dialects/postgres"
// import _ "github.com/jinzhu/gorm/dialects/sqlite"
// import _ "github.com/jinzhu/gorm/dialects/mssql"
所以包名可以改为如上:
import (
"github.com/jinzhu/gorm"
_ "github.com/jinzhu/gorm/dialects/mysql"
)
func main() {
db, err := gorm.Open("mysql", "user:password@tcp(IP:port)/dbname?charset=utf8&parseTime=True&loc=Local")
db.DB().SetMaxIdleConns(10)
db.DB().SetMaxOpenConns(100)
defer db.Close()
}
注:为了处理time.Time,你需要包括parseTime作为参数。
数据模型定义
表名,列名如何对应结构体
在Gorm中,表名是结构体名的复数形式,列名是字段名的蛇形小写。
即,如果有一个user表,那么如果你定义的结构体名为:User,gorm会默认表名为users而不是user。
例如有如下表结构定义:
CREATE TABLE `areas` (
`id` int(11) NOT NULL AUTO_INCREMENT COMMENT '主键id',
`area_id` int(11) NOT NULL COMMENT '区县id',
`area_name` varchar(45) NOT NULL COMMENT '区县名',
`city_id` int(11) NOT NULL COMMENT '城市id',
`city_name` varchar(45) NOT NULL COMMENT '城市名称',
`province_id` int(11) NOT NULL COMMENT '省份id',
`province_name` varchar(45) NOT NULL COMMENT '省份名称',
`area_status` tinyint(3) NOT NULL DEFAULT '1' COMMENT '该条区域信息是否可用 : 1:可用 2:不可用',
`created_at` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',
`updated_at` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '更新时间',
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8mb4 COMMENT='区域表'
那么对应的结构体定义如下:
type Area struct {
Id int
AreaId int
AreaName string
CityId int
CityName string
ProvinceId int
ProvinceName string
AreaStatus int
CreatedAt time.Time
UpdatedAt time.Time
}
如何全局禁用表名复数呢?
可以在创建数据库连接的时候设置如下参数:
// 全局禁用表名复数
db.SingularTable(true) // 如果设置为true,`User`的默认表名为`user`,使用`TableName`设置的表名不受影响
这样的话,表名默认即为结构体的首字母小写形式。
CRUD 使用
下面我们使用一张User表来就CRUD做一些操作示例:
表结构如下:
CREATE TABLE `user` (
`id` bigint(20) NOT NULL AUTO_INCREMENT,
`name` varchar(30) NOT NULL DEFAULT '',
`age` int(3) NOT NULL DEFAULT '0',
`sex` tinyint(3) NOT NULL DEFAULT '0',
`phone` varchar(40) NOT NULL DEFAULT '',
`create_at` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
`updated_at` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP,
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=8 DEFAULT CHARSET=utf8mb4
首先初始化数据库连接:
package db2
import (
"fmt"
"github.com/jinzhu/gorm"
_ "github.com/jinzhu/gorm/dialects/mysql"
"time"
)
var db *gorm.DB
type User struct {
gorm.Model
Name string
Age int
Birthday time.Time
}
func TestInitTable1() *gorm.DB {
databaseType := "mysql"
username := "root" //账号
password := "123456" //密码
host := "192.168.11.12" //数据库地址,可以是Ip或者域名
port := 3306 //数据库端口
Dbname := "flink" //数据库名
timeout := "10s" //连接超时,10秒
dsn := fmt.Sprintf("%s:%s@tcp(%s:%d)/%s?charset=utf8&parseTime=True&loc=Local&timeout=%s", username, password, host, port, Dbname, timeout)
//拼接下dsn参数, dsn格式可以参考上面的语法,这里使用Sprintf动态拼接dsn参数,因为一般数据库连接参数,我们都是保存在配置文件里面,需要从配置文件加载参数,然后拼接dsn。
//使用gorm链接数据库
db, err := gorm.Open(databaseType, dsn)
if err != nil {
fmt.Println("数据库链接失败", err) //数据库链接失败是致命的错误,链接失败后可以关闭程序了,所以使用logging.Fatal方法
}
//设置全局表名禁用复数
db.SingularTable(true)
db.AutoMigrate(&User{})
return db
}
func TestInitTable2() *gorm.DB{
databaseType := "mysql"
username := "root" //账号
password := "123456" //密码
host := "192.168.11.12" //数据库地址,可以是Ip或者域名
port := 3306 //数据库端口
Dbname := "flink" //数据库名
timeout := "10s" //连接超时,10秒
dsn := fmt.Sprintf("%s:%s@tcp(%s:%d)/%s?charset=utf8&parseTime=True&loc=Local&timeout=%s", username, password, host, port, Dbname, timeout)
//拼接下dsn参数, dsn格式可以参考上面的语法,这里使用Sprintf动态拼接dsn参数,因为一般数据库连接参数,我们都是保存在配置文件里面,需要从配置文件加载参数,然后拼接dsn。
//使用gorm链接数据库
db, err := gorm.Open(databaseType, dsn)
if err != nil {
fmt.Println("数据库链接失败", err) //数据库链接失败是致命的错误,链接失败后可以关闭程序了,所以使用logging.Fatal方法
}
//设置全局表名禁用复数
db.SingularTable(true)
db.LogMode(true) //打印日志,本地调试的时候可以打开看执行的sql语句
db.DB().SetMaxIdleConns(10) //设置空闲时的最大连接数
db.DB().SetMaxOpenConns(100) //设置数据库的最大打开连接数
db.AutoMigrate(&User{})
return db
}
下面所有的操作都是在上面的初始化连接上执行的操作。
插入
//插入数据
func (user *User) Insert() {
db = TestInitTable1()
//fmt.Println(db)
//这里使用了Table()函数,如果你没有指定全局表名禁用复数,或者是表名跟结构体名不一样的时候
//你可以自己在sql中指定表名。这里是示例,本例中这个函数可以去除。
//db.Table("user").Create(user)
db.Table("user").Debug().Create(user)
}
func TestCreateUser() {
//新增数据
for i := 0; i < 10; i++ {
user := User{
Name: fmt.Sprintf("%s%d", "wang", i),
Age: 31 + i,
Birthday: time.Now(),
}
//fmt.Println(user)
user.Insert()
}
}
更新
func TestUpdateUser() {
db = TestInitTable1()
user := User{Name: "xiaoming", Age: 18, Birthday: time.Now()}
user.ID = 1
db.Model(&user).Debug().Update(user)
//注意到上面Update中使用了一个Struct,你也可以使用map对象。
//需要注意的是:使用Struct的时候,只会更新Struct中这些非空的字段。
//对于string类型字段的"",int类型字段0,bool类型字段的false都被认为是空白值,不会去更新表
//下面这个更新操作只使用了where条件没有在Model中指定id
update user set name='xiaohong' wehre sex=1
db.Model(&User{}).Where("id = ?", 2).Update("name", "wangwu").Update("age", 30)
var user User
user.ID = 3
db.Model(&user).Select("name").Update(map[string]interface{}{"name": "lisi", "age": 30})
}
如果你想手动将某个字段set为空值, 可以使用单独选定某些字段的方式来更新:
user := User{Name: "wangwu", Age: 20}
user.ID = 4
db.Model(&user).Omit("name").Update(&user)
忽略掉某些字段:
当你的更新的参数为结构体,而结构体中某些字段你又不想去更新,那么可以使用Omit方法过滤掉这些不想update到库的字段:
user := User{Id: 1,Name:"xioaming",Age:12}
db.Model(&user).Omit("name").Update(&user)
删除
func TestDelete() {
fmt.Println("* TestDelete ")
db = TestInitTable1()
//delete from user where id=1;
//var user User
//user.ID = 1
//db.Delete(&user)
//db.Delete(&User{}, 2)
//批量删除
db.Where("name LIKE ?", "%lisi%").Delete(User{})
db.Exec("DELETE FROM user WHERE id <= 5")
}
事务
func CreateAnimals(db *gorm.DB) err {
tx := db.Begin()
// 注意,一旦你在一个事务中,使用tx作为数据库句柄
if err := tx.Create(&Animal{Name: "Giraffe"}).Error; err != nil {
tx.Rollback()
return err
}
if err := tx.Create(&Animal{Name: "Lion"}).Error; err != nil {
tx.Rollback()
return err
}
tx.Commit()
return nil
}
查询
例子1
func (user *User) query() (u []User) {
//查询所有记录
db.Find(&u)
//Find方法可以带 where 参数
db.Find(&u,"id > ? and age > ?",2,12)
//带where 子句的查询,注意where要在find前面
db.Where("id > ?", 2).Find(&u)
// where name in ("xiaoming","xiaohong")
db.Where("name in (?)",[]string{"xiaoming","xiaohong"}).Find(&u)
//获取第一条记录,按照主键顺序排序
db.First(&u)
//First方法可以带where 条件
db.First(&u,"where sex = ?",1)
//获取最后一条记录,按照主键顺序排序
//同样 last方法也可以带where条件
db.Last(&u)
return u
}
注意:方法中带的&u表示是返回值用u这个对象来接收。
例子2
func TestSelect() {
db = TestInitTable1()
// 查询全部数据
//users := []User{}
//db.Debug().Find(&users)
//
//for idx, user := range users {
// fmt.Println(idx, user.ID)
//}
// 查询指定字段的记录
users := []User{}
//指定查询字段
db.Debug().Select("name,age").Where(map[string]interface{}{"age": 36, "name": "wang5"}).Find(&users)
for idx, user := range users {
fmt.Println(idx, user.ID, user.Name, user.Age)
}
//fmt.Println(users)
}
上面的查询都将返回表中所有的字段,如果你想指定查询某些字段该怎么做呢?
指定查询字段-Select
/指定查询字段
db.Select("name,age").Where(map[string]interface{}{"age":12,"sex":1}).Find(&u)
使用Struct和map作为查询条件
/使用Struct,相当于:select * from user where age =12 and sex=1
db.Where(&User{Age:12,Sex:1}).Find(&u)
//等同上一句
db.Where(map[string]interface{}{"age":12,"sex":1}).Find(&u)
not 条件的使用
//where name not in ("xiaoming","xiaohong")
db.Not("name","xiaoming","xiaohong").Find(&u)
//同上
db.Not("name",[]string{"xiaoming","xiaohong"}).Find(&u)
or 的使用
//where age > 12 or sex = 1
db.Where("age > ?",12).Or("sex = ?",1).Find(&u)
order by 的使用
//order by age desc
db.Where("age > ?",12).Or("sex = ?",1).Order("age desc").Find(&u)
limit 的使用
//limit 10
db.Not("name",[]string{"xiaoming","xiaohong"}).Limit(10).Find(&u)
offset 的使用
//limit 300,10
db.Not("name",[]string{"xiaoming","xiaohong"}).Limit(10).Offset(300).Find(&u)
count(*)
//count(*)
var count int
db.Table("user").Where("age > ?",0).Count(&count)
注意:这里你在指定表名的情况下sql为:select count(*) from user where age > 0;
如上代码如果改为:
var count int
var user []User
db.Where("age > ?",0).Find(&user).Count(&count)
相当于你先查出来[]User,然后统计这个list的长度。跟你预期的sql不相符。
group & having
rows, _ := db.Table("user").Select("count(*),sex").Group("sex").
Having("age > ?", 10).Rows()
for rows.Next() {
fmt.Print(rows.Columns())
}
join
db.Table("user u").Select("u.name,u.age").Joins("left join user_ext ue on u.user_id = ue.user_id").Row()
如果有多个连接,用多个Join方法即可。
原生函数
db.Exec("DROP TABLE user;")
db.Exec("UPDATE user SET name=? WHERE id IN (?)", "xiaoming", []int{11,22,33})
db.Exec("select * from user where id > ?",10).Scan(&user)
一些函数
FirstOrInit 和 FirstOrCreate
获取第一个匹配的记录,若没有,则根据条件初始化一个新的记录:
//注意:where条件只能使用Struct或者map。如果这条记录不存在,那么会新增一条name=xiaoming的记录
db.FirstOrInit(&u,User{Name:"xiaoming"})
//同上
db.FirstOrCreate(&u,User{Name:"xiaoming"})
Attrs
如果没有找到记录,则使用Attrs中的数据来初始化一条记录:
/使用attrs来初始化参数,如果未找到数据则使用attrs中的数据来初始化一条
//注意:attrs 必须 要和FirstOrInit 或者 FirstOrCreate 连用
db.Where(User{Name:"xiaoming"}).Attrs(User{Name:"xiaoming",Age:12}).FirstOrInit(&u)
Assign
//不管是否找的到,最终返回结构中都将带上Assign指定的参数
db.Where("age > 12").Assign(User{Name:"xiaoming"}).FirstOrInit(&u)
Pluck
如果user表中你只想查询age这一列,该怎么返回呢,gorm提供了Pluck函数用于查询单列,返回数组:
//不管是否找的到,最终返回结构中都将带上Assign指定的参数
db.Where("age > 12").Assign(User{Name:"xiaoming"}).FirstOrInit(&u)
Scan
Scan函数可以将结果转存储到另一个结构体中。
type SubUser struct{
Name string
Age int
}
db.Table("user").Select("name,age").Scan(&SubUser)
sql.Row & sql.Rows
row和rows用户获取查询结果。
//查询一行
row := db.Table("user").Where("name = ?", "xiaoming").Select("name, age").Row() // (*sql.Row)
//获取一行的结果后,调用Scan方法来将返回结果赋值给对象或者结构体
row.Scan(&name, &age)
//查询多行
rows, err := db.Model(&User{}).Where("sex = ?",1).Select("name, age, phone").Rows() // (*sql.Rows, error)
defer rows.Close()
for rows.Next() {
...
rows.Scan(&name, &age, &email)
...
}
日志
Gorm有内置的日志记录器支持,默认情况下,它会打印发生的错误。
// 启用Logger,显示详细日志
db.LogMode(true)
// 禁用日志记录器,不显示任何日志
db.LogMode(false)
// 调试单个操作,显示此操作的详细日志
db.Debug().Where("name = ?", "xiaoming").First(&User{})