# 准备工作 推荐使用[kratos工具](kratos-tool.md)快速生成项目,如我们生成一个叫`kratos-demo`的项目。目录结构如下: ``` ├── CHANGELOG.md ├── CONTRIBUTORS.md ├── LICENSE ├── README.md ├── cmd │   ├── cmd │   └── main.go ├── configs │   ├── application.toml │   ├── grpc.toml │   ├── http.toml │   ├── log.toml │   ├── memcache.toml │   ├── mysql.toml │   └── redis.toml ├── go.mod ├── go.sum └── internal ├── dao │   └── dao.go ├── model │   └── model.go ├── server │   └── http │   └── http.go └── service └── service.go ``` # 开始使用 ## 配置 创建项目成功后,进入项目中的configs目录,mysql.toml,我们可以看到: ```toml [demo] addr = "127.0.0.1:3306" dsn = "{user}:{password}@tcp(127.0.0.1:3306)/{database}?timeout=1s&readTimeout=1s&writeTimeout=1s&parseTime=true&loc=Local&charset=utf8mb4,utf8" readDSN = ["{user}:{password}@tcp(127.0.0.2:3306)/{database}?timeout=1s&readTimeout=1s&writeTimeout=1s&parseTime=true&loc=Local&charset=utf8mb4,utf8","{user}:{password}@tcp(127.0.0.3:3306)/{database}?timeout=1s&readTimeout=1s&writeTimeout=1s&parseTime=true&loc=Local&charset=utf8,utf8mb4"] active = 20 idle = 10 idleTimeout ="4h" queryTimeout = "200ms" execTimeout = "300ms" tranTimeout = "400ms" ``` 在该配置文件中我们可以配置mysql的读和写的dsn、连接地址addr、连接池的闲置连接数idle、最大连接数active以及各类超时。 如果配置了readDSN,在进行读操作的时候会优先使用readDSN的连接。 ## 初始化 进入项目的internal/dao目录,打开dao.go,其中: ```go var ( dc struct { Demo *sql.Config } ) checkErr(paladin.Get("mysql.toml").UnmarshalTOML(&dc)) ``` 使用paladin配置管理工具将上文中的mysql.toml中的配置解析为我们需要使用mysql的相关配置。 ```go // Dao dao. type Dao struct { db *sql.DB } ``` 在dao的主结构提中定义了mysql的连接池对象。 ```go dao = &Dao{ db: sql.NewMySQL(dc.Demo), } ``` 使用kratos/pkg/database/sql包的NewMySQL方法进行连接池对象的初始化,需要传入上文解析的配置。 ## Ping ```go // Ping ping the resource. func (d *Dao) Ping(ctx context.Context) (err error) { return d.db.Ping(ctx) } ``` 生成的dao层模板中自带了mysql相关的ping方法,用于为负载均衡服务的健康监测提供依据,详见[blademaster](blademaster-quickstart.md)。 ## 关闭 ```go // Close close the resource. func (d *Dao) Close() { d.db.Close() } ``` 在关闭dao层时,通过调用mysql连接池对象的Close方法,我们可以关闭该连接池,从而释放相关资源。 # 常用方法 ## 单个查询 ```go // GetUserRole 用户角色 func (d *Dao) GetDemo(c context.Context, did int64) (demo int8, err error) { err = d.db.QueryRow(c, _getDemoSQL, did).Scan(&demo) if err != nil && err != sql.ErrNoRows { log.Error("d.managerDB.Query error(%v)", err) return } return demo, nil } ``` db.QueryRow方法用于返回最多一条记录的查询,在QueryRow方法后使用Scan方法即可将mysql的返回值转换为Golang的数据类型。 ## 批量查询 ```go // ResourceLogs ResourceLogs. func (d *Dao) GetDemos(c context.Context, dids []int64) (demos []int8, err error) { rows, err := d.db.Query(c, _getDemosSQL, dids) if err != nil { log.Error("query error(%v)", err) return } defer rows.Close() for rows.Next() { var tmpD int8 if err = rows.Scan(&tmpD); err != nil { log.Error("scan demo log error(%v)", err) return } demos = append(demos, tmpD) } return } ``` db.Query方法一般用于批量查询的场景,返回*sql.Rows和error信息。 我们可以使用rows.Next()方法获得下一行的返回结果,并且配合使用rows.Scan()方法将该结果转换为Golang的数据类型。当没有下一行时,rows.Next方法将返回false,此时循环结束。 注意,在使用完毕rows对象后,需要调用rows.Close方法关闭连接,释放相关资源。 ## 执行语句 ```go // DemoExec exec func (d *Dao) DemoExec(c context.Context, id int64) (rows int64, err error) { res, err := d.db.Exec(c, _demoUpdateSQL, id) if err != nil { log.Error("db.DemoExec.Exec(%s) error(%v)", _demoUpdateSQL, err) return } return res.RowsAffected() } ``` 执行UPDATE/DELETE/INSERT语句时,使用db.Exec方法进行语句执行,返回*sql.Result和error信息: ```go // A Result summarizes an executed SQL command. type Result interface { LastInsertId() (int64, error) RowsAffected() (int64, error) } ``` Result接口支持获取影响行数和LastInsertId(一般用于获取Insert语句插入数据库后的主键ID) ## 事务 kratos/pkg/database/sql包同样支持事务操作。 开启一个事务: ```go tx := d.db.Begin() if err = tx.Error; err != nil { log.Error("db begin transcation failed, err=%+v", err) return } ``` 在事务中执行语句: ```go res, err := tx.Exec(_demoSQL, did) if err != nil { return } rows := res.RowsAffected() ``` 提交事务: ```go if err = tx.Commit().Error; err!=nil{ log.Error("db commit transcation failed, err=%+v", err) } ``` 回滚事务: ```go if err = tx.Rollback().Error; err!=nil{ log.Error("db rollback failed, err=%+v", rollbackErr) } ``` # 扩展阅读 [tidb模块说明](database-tidb.md) [hbase模块说明](database-hbase.md)