1
0
mirror of https://github.com/go-kratos/kratos.git synced 2025-01-24 03:46:37 +02:00
kratos/doc/wiki-cn/database-mysql.md
zhaoshichen 61c85829d1 add
2019-06-09 10:57:37 +08:00

5.5 KiB

准备工作

推荐使用kratos工具快速生成项目,如我们生成一个叫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,我们可以看到:

[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,其中:

    var (
		dc struct {
			Demo *sql.Config
		}
    )
	checkErr(paladin.Get("mysql.toml").UnmarshalTOML(&dc))

使用paladin配置管理工具将上文中的mysql.toml中的配置解析为我们需要使用mysql的相关配置。

// Dao dao.
type Dao struct {
	db          *sql.DB
}

在dao的主结构提中定义了mysql的连接池对象。

	dao = &Dao{
        db: sql.NewMySQL(dc.Demo),
	}

使用kratos/pkg/database/sql包的NewMySQL方法进行连接池对象的初始化,需要传入上文解析的配置。

Ping

// Ping ping the resource.
func (d *Dao) Ping(ctx context.Context) (err error) {
	return d.db.Ping(ctx)
}

生成的dao层模板中自带了mysql相关的ping方法,用于为负载均衡服务的健康监测提供依据,详见blademaster

关闭

// Close close the resource.
func (d *Dao) Close() {
	d.db.Close()
}

在关闭dao层时,通过调用mysql连接池对象的Close方法,我们可以关闭该连接池,从而释放相关资源。

常用方法

单个查询

// 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的数据类型。

批量查询

// 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方法关闭连接,释放相关资源。

执行语句

// 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信息:


// A Result summarizes an executed SQL command.
type Result interface {
	LastInsertId() (int64, error)
	RowsAffected() (int64, error)
}

Result接口支持获取影响行数和LastInsertId(一般用于获取Insert语句插入数据库后的主键ID)

事务

kratos/pkg/database/sql包同样支持事务操作。

开启一个事务:

	tx := d.db.Begin()
	if err = tx.Error; err != nil {
		log.Error("db begin transcation failed, err=%+v", err)
		return
	}

在事务中执行语句:

    res, err := tx.Exec(_demoSQL, did)
	if err != nil {
		return
	}
	rows := res.RowsAffected()

提交事务:

    if err = tx.Commit().Error; err!=nil{
        log.Error("db commit transcation failed, err=%+v", err)
    }

回滚事务:

    if err = tx.Rollback().Error; err!=nil{
    	log.Error("db rollback failed, err=%+v", rollbackErr)
    }

扩展阅读

tidb模块说明 hbase模块说明