golang常用库:gorilla/mux-http路由库使用
golang常用库:配置文件解析库-viper使用
golang常用库:操作数据库的orm框架-gorm基本使用
golang常用库:字段参数验证库-validator使用

一:字段映射-模型定义

gorm中通常用struct来映射字段. gorm教程中叫模型定义

比如我们定义一个模型Model:

  1. type User struct {
  2. gorm.Model
  3. UserId int64 `gorm:"index"` //设置一个普通的索引,没有设置索引名,gorm会自动命名
  4. Birtheday time.Time
  5. Age int `gorm:"column:age"`//column:一个tag,可以设置列名称
  6. Name string `gorm:"size:255;index:idx_name_add_id"`//size:设置长度大小,index:设置索引,这个就取了一个索引名
  7. Num int `gorm:"AUTO_INCREMENT"`
  8. Email string `gorm:"type:varchar(100);unique_index"`//type:定义字段类型和大小
  9. AddressID sql.NullInt64 `gorm:"index:idx_name_add_id"`
  10. IgnoreMe int `gorm:"_"`
  11. Description string `gorm:"size:2019;comment:'用户描述字段'"`//comment:字段注释
  12. Status string `gorm:"type:enum('published', 'pending', 'deleted');default:'pending'"`
  13. }

上面的gorm.Model 定义如下:

  1. type Model struct {
  2. ID uint `gorm:"primary_key"`//primary_key:设置主键
  3. CreatedAt time.Time
  4. UpdatedAt time.Time
  5. DeletedAt *time.Time
  6. }

当然我们也可以不用gorm.Model,自己定义一个差不多的类型

如果你用ID,系统会自动设为表的主键,当然我们可以自己定义主键:
比如:

  1. // 使用`AnimalID`作为主键
  2. type Animal struct {
  3. AnimalID int64 `gorm:"primary_key"`
  4. Name string
  5. Age int64
  6. }

参考:https://gorm.io/zh_CN/docs/conventions.html

二:创建表

直接看下面的例子:createtable.go

  1. package main
  2. import (
  3. "database/sql"
  4. "fmt"
  5. "github.com/jinzhu/gorm"
  6. _ "github.com/jinzhu/gorm/dialects/mysql"
  7. "time"
  8. )
  9. type User struct {
  10. gorm.Model
  11. UserId int64 `gorm:"index"`
  12. Birtheday time.Time
  13. Age int `gorm:"column:age"`
  14. Name string `gorm:"size:255;index:idx_name_add_id"`
  15. Num int `gorm:"AUTO_INCREMENT"`
  16. Email string `gorm:"type:varchar(100);unique_index"`
  17. AddressID sql.NullInt64 `gorm:"index:idx_name_add_id"`
  18. IgnoreMe int `gorm:"_"`
  19. Description string `gorm:"size:2019;comment:'用户描述字段'"`
  20. Status string `gorm:"type:enum('published', 'pending', 'deleted');default:'pending'"`
  21. }
  22. //设置表名,默认是结构体的名的复数形式
  23. func (User) TableName() string {
  24. return "VIP_USER"
  25. }
  26. func main() {
  27. db, err := gorm.Open("mysql", "root:root@/gormdemo?charset=utf8&parseTime=True&loc=Local")
  28. if err != nil {
  29. fmt.Println("connect db err: ", err)
  30. }
  31. defer db.Close()
  32. if db.HasTable(&User{}) { //判断表是否存在
  33. db.AutoMigrate(&User{}) //存在就自动适配表,也就说原先没字段的就增加字段
  34. } else {
  35. db.CreateTable(&User{}) //不存在就创建新表
  36. }
  37. }

上面的gorm.Open()操作,如果想指定主机话,就需要加上括号 ()
例如:
user:password@(localhost)/dbname?charset=utf8&parseTime=True&loc=Local

上面的程序中,先新建了一个数据库名叫 gormdemo,然后运行:go run createtable.go , 成功运行后,数据库就会出现一张名为 vip_user 的表。

三:增删改查

新建一个gormdemo的数据库,然后执行下面的sql语句,就会建立一个animals的表,里面还有一些测试数据

  1. CREATE TABLE `animals` (
  2. `id` bigint(20) NOT NULL AUTO_INCREMENT,
  3. `name` varchar(255) DEFAULT 'galeone',
  4. `age` int(10) unsigned DEFAULT '0',
  5. PRIMARY KEY (`id`)
  6. ) ENGINE=MyISAM AUTO_INCREMENT=8 DEFAULT CHARSET=utf8;
  7. -- ----------------------------
  8. -- Records of animals
  9. -- ----------------------------
  10. INSERT INTO `animals` VALUES ('1', 'demo-test', '20');
  11. INSERT INTO `animals` VALUES ('2', 'galeone', '30');
  12. INSERT INTO `animals` VALUES ('3', 'demotest', '30');
  13. INSERT INTO `animals` VALUES ('4', 'jim', '90');
  14. INSERT INTO `animals` VALUES ('5', 'jimmy', '10');
  15. INSERT INTO `animals` VALUES ('6', 'jim', '23');
  16. INSERT INTO `animals` VALUES ('7', 'test3', '27');

增加

例子:create.go

  1. package main
  2. import (
  3. "fmt"
  4. "github.com/jinzhu/gorm"
  5. _ "github.com/jinzhu/gorm/dialects/mysql"
  6. )
  7. type Animal struct {
  8. ID int64
  9. Name string
  10. Age int64
  11. }
  12. func main() {
  13. db, err := gorm.Open("mysql", "root:root@/gormdemo?charset=utf8&parseTime=true&loc=Local")
  14. if err != nil {
  15. fmt.Println("connect db error: ", err)
  16. }
  17. defer db.Close()
  18. animal := Animal{Name: "demo-test", Age: 20}
  19. db.Create(&animal)
  20. }

说明:上面的这个例子,自己在mysql中创建一个animals的数据表,字段为id,name,age

查找

select.go

  1. package main
  2. import (
  3. "fmt"
  4. "github.com/jinzhu/gorm"
  5. _ "github.com/jinzhu/gorm/dialects/mysql"
  6. )
  7. type Animal struct {
  8. ID int64
  9. Name string
  10. Age int64
  11. }
  12. //https://gorm.io/zh_CN/docs/query.html
  13. func main() {
  14. db, err := gorm.Open("mysql", "root:root@/gormdemo?charset=utf8&parseTime=true&loc=Local")
  15. if err != nil {
  16. fmt.Println("connect db error: ", err)
  17. }
  18. defer db.Close()
  19. //根据逐渐查询第一条记录
  20. var animal Animal
  21. db.First(&animal)
  22. fmt.Println(animal)
  23. //根据逐渐查询最后一条记录
  24. var animal2 Animal
  25. db.Last(&animal2)
  26. fmt.Println(animal2)
  27. //指定某条记录(仅当主键为整型时可用)
  28. var animal3 Animal
  29. db.First(&animal3, 2)
  30. fmt.Println(animal3)
  31. //where条件
  32. //符合条件的第一条记录
  33. var animal4 Animal
  34. db.Where("name = ?", "demotest2").First(&animal4)
  35. fmt.Println("where : ", animal4, animal4.ID, animal4.Name, animal4.Age)
  36. //符合条件的所有记录
  37. var animals5 []Animal
  38. db.Where("name = ?", "galeone").Find(&animals5)
  39. fmt.Println(animals5)
  40. for k, v := range animals5 {
  41. fmt.Println("k:", k, "ID:", v.ID, "Name:", v.Name, "Age:", v.Age)
  42. }
  43. //IN
  44. var animals6 []Animal
  45. db.Where("name IN (?)", []string{"demo-test", "demotest2"}).Find(&animals6)
  46. fmt.Println(animals6)
  47. //LIKE
  48. var animals7 []Animal
  49. db.Where("name like ?", "%jim%").Find(&animals7)
  50. fmt.Println(animals7)
  51. //AND
  52. var animals8 []Animal
  53. db.Where("name = ? AND age >= ?", "jim", "24").Find(&animals8)
  54. fmt.Println(animals8)
  55. //总数
  56. var count int
  57. var animals9 []Animal
  58. db.Where("name = ?", "galeone").Or("name = ?", "jim").Find(&animals9).Count(&count)
  59. fmt.Println(animals9)
  60. fmt.Println(count)
  61. //Scan, 原生查询
  62. var animals10 []Animal
  63. db.Raw("SELECT id, name, age From Animals WHERE name = ? AND age = ? ", "galeone", "30").Scan(&animals10)
  64. fmt.Println("Scan: ", animals10)
  65. //原生查询,select all
  66. var animals11 []Animal
  67. rows, _ := db.Raw("SELECT id,name FROM Animals").Rows()
  68. //注意:上面的 select id,name 后面不能写成 * 代替,不然出来的结果都是默认0值
  69. //像这样结果: ALL: [{0 0} {0 0} {0 0} {0 0} {0 0} {0 0} {0 0}]
  70. //Scan 后面是什么字段,select 后面就紧跟什么字段
  71. for rows.Next() {
  72. var result Animal
  73. rows.Scan(&result.ID, &result.Name)
  74. animals11 = append(animals11, result)
  75. }
  76. fmt.Println("ALL: ", animals11)
  77. //output:ALL: [{1 demo-test 0} {2 galeone 0} {3 demotest2 0} {4 galeone 0} {5 galeone 0} {6 jim 0} {7 jimmy 0}]
  78. //select 查询
  79. var animal12 Animal
  80. db.Select("name,age").Find(&animal12) //只查询name,age字段,相当于select name,age from user
  81. fmt.Println("select: ", animal12)
  82. // db.Select([]string{"name", "age"}).Find(&animal12)
  83. // fmt.Println("select2: ", animal12)
  84. }

更新

update.go

  1. package main
  2. import (
  3. "fmt"
  4. "github.com/jinzhu/gorm"
  5. _ "github.com/jinzhu/gorm/dialects/mysql"
  6. )
  7. type Animal struct {
  8. ID int64
  9. Name string
  10. Age int64
  11. }
  12. func main() {
  13. db, err := gorm.Open("mysql", "root:root@/gormdemo?charset=utf8&parseTime=true&loc=Local")
  14. if err != nil {
  15. fmt.Println("connect db error: ", err)
  16. }
  17. defer db.Close()
  18. ///根据一个条件更新
  19. //根据条件更新字段值,
  20. //后面加Debug(),运行时,可以打印出sql
  21. db.Debug().Model(&Animal{}).Where("id = ? ", 4).Update("name", "jimupdate")
  22. //UPDATE `animals` SET `name` = 'jimupdate' WHERE (id = 4)
  23. //另外一种写法: 根据条件更新
  24. var animal Animal
  25. animal = Animal{ID: 3}
  26. db.Debug().Model(animal).Update("name", "demotest2update")
  27. // db.Debug().Model(&animal).Update("name", "demotest2update") // 这种写法也可以
  28. //UPDATE `animals` SET `name` = 'demotest2update' WHERE `animals`.`id` = 3
  29. /// 多个条件更新
  30. db.Model(&Animal{}).Where("id = ? AND age = ?", 4, 45).Update("name", "jimupdate3")
  31. //UPDATE `animals` SET `name` = 'jimupdate2' WHERE (id = 4 AND age = 45)
  32. /// 更新多个值
  33. db.Debug().Model(&Animal{}).Where("id = ?", 4).Update(Animal{Name: "jim", Age: 90})
  34. // UPDATE `animals` SET `age` = 90, `name` = 'jim' WHERE (id = 4)
  35. animal2 := Animal{ID: 5}
  36. db.Debug().Model(&animal2).Update(map[string]interface{}{"name": "jimm", "age": 100})
  37. //UPDATE `animals` SET `age` = 100, `name` = 'jimm' WHERE `animals`.`id` = 5
  38. }

删除

delete.go

  1. package main
  2. import (
  3. "fmt"
  4. "github.com/jinzhu/gorm"
  5. _ "github.com/jinzhu/gorm/dialects/mysql"
  6. )
  7. type Animal struct {
  8. ID int64
  9. Name string
  10. Age int64
  11. }
  12. func main() {
  13. db, err := gorm.Open("mysql", "root:root@/gormdemo?charset=utf8&parseTime=true&loc=Local")
  14. if err != nil {
  15. fmt.Println("connect db error: ", err)
  16. }
  17. defer db.Close()
  18. db.Debug().Where("id = ?", 13).Delete(&Animal{})
  19. // DELETE FROM `animals` WHERE (id = 13)
  20. db.Debug().Delete(&Animal{}, "id = ? AND age = ?", 14, 10)
  21. //DELETE FROM `animals` WHERE (id = 14 AND age = 10)
  22. }

四:Debug

在db后面直接加上 Debug(), 比如delete.go 里面的例子

五:参考

https://gorm.io/zh_CN/