golang-mysql、sqlx

1.导包

go get -u github.com/go-sql-driver/mysql
import _ "github.com/go-sql-driver/mysql"

_ 表示只执行包中init函数,mysql包会在init函数中注册自己。

2.连接数据库

利用基本库database/sql连接数据库

	dsn := "root:123456@tcp(127.0.0.1:3306)/test_db"
	db, err := sql.Open("mysql", dsn)
	db.SetMaxIdleConns(10) //设立连接池的最大连接数
	if err != nil {
		panic(err)
	}
	defer db.Close()

3.CRUD操作

常用函数:

func (db *DB) Query(query string, args …any) (*Rows, error)
func (db *DB) QueryRow(query string, args …any) *Row //查询单行
func (db *DB) Exec(query string, args …any) (Result, error)

Query和Exec的主要区别是Query(查询)能够知道返回的结果集,而Exec(执行)只能知道最后插入的ID和影响的行数。

查询

	rows, err := db.Query("select * from test_db.user_table")
	for rows.Next() {
		var id, age int
		var name string
		err := rows.Scan(&id, &name, &age)
		if err != nil {
			panic(err)
		}
		fmt.Println(id, name, age)
	}

查询单行:

	var id, age int
	var name string
	_ = db.QueryRow("select * from test_db.user_table").Scan(&id, &name, &age)
	println(id, name, age)

插入

	result, err := db.Exec("insert into test_db.user_table values (?,?,?)", 4, "gi", 19)
	if err != nil {
		panic(err)
	}
	id, _ := result.LastInsertId()
	println("lastInsetId:", id)

更新,删除与插入类似,不多叙述。

事务

func Transaction(db *sql.DB) {
	//事务开启
	tx, err := db.Begin()
	if err != nil {
		tx.Rollback()
		log.Println(err)
		return
	}
	_, err = tx.Exec("update online_info set Status=? where User_id=?", "online", 1)
	if err != nil {
		tx.Rollback()
		log.Println(err)
		return
	}
	//提交事务
	err = tx.Commit()
	if err != nil {
		tx.Rollback()
		log.Println(err)
		return
	}
}

sqlx库

sqlx是sql的超集,能够兼容sql库的所有方法,包括db.Exec\Query\Queryrow等等。

sqlx特性

db.Get():查询一行

在查询的时候将结果通过反射赋值给结构体。
注意:需要传入结构体实例的地址,结构体字段要大写!

	db, err := sqlx.Open("mysql", dsn)
	if err != nil {
		panic(err)
	}
	var a account  
	err= db.Get(&a,"select * from online_info where User_id=?", 1)
db.Select():查询多行结果

dest需要为slice类型的指针,

	var a []account = make([]account, 0)
	err = db.Select(&a, "select * from online_info where User_id=?", 1)
	if err != nil {
		panic(err)
	}
	fmt.Println(a)
namedQuery():将args参数绑定在sql中的:arg字段
rows, err := db.NamedQuery("select * from online_info where User_id=:id", map[string]interface{}{
		"id": 1
})
namedExec():与namedQuery作用相同,用于Exec()
StructScan():Scan增强版,直接将结果中的值赋给结构体变量。

连接池

源码分析:https://zhuanlan.zhihu.com/p/430993402

连接池配置

DB.SetMaxIdleConns(n int) 设置连接池中的空闲连接的最大连接数。默认也是0,表示连接池不会保持数据库连接的状态:即当连接释放回到连接池的时候,连接将会被关闭。这会导致连接再连接池中频繁的关闭和创建,我们可以设置一个合理的值。

DB.SetMaxOpenConns(n int) 设置打开数据库的最大连接数。包含正在使用的连接和连接池的连接。如果你的方法调用 需要用到一个连接,并且连接池已经没有了连接或者连接数达到了最大连接数。此时的方法调用将会被 block,直到有可用的连接才会返回。设置这个值可以避免并发太高导致连接 mysql 出现 too many connections 的错误。该函数的默认设置是0,表示无限制

DB.SetConnMaxLifetime(d time.Duration) 设置连接可以被使用的最长有效时间,如果过期,连接将被拒绝

总结

只用 sqlx.Open() 函数创建连接池,此时只是初始化了连接池,并没有连接数据库,连接都是惰性的,只有调用 sqlx.DB 的方法时,此时才真正用到了连接,连接池才会去创建连接,连接池很重要,它直接影响着你的程序行为。

连接池的工作原理也非常简单,当调用 sqlx.DB 的方法时,会首先去向连接池请求要一个数据库连接,如果连接池有空闲的连接,则返回给方法中使用,否则连接池将创建一个新的连接给到方法中使用;一旦将数据库连接给到了方法中,连接就属于方法了。方法执行完毕后,要不把连接所属权还给连接池,要不传递给下一个需要数据库连接的方法中,最后都使用完将连接释放回到连接池中

请求数据库连接的方法有几个,执行完毕处理连接的方式也不同:

  1. DB.Ping() 使用完毕后会马上把连接返回给连接池 DB.Exec() 使用完毕后会马上把连接返回给连接池,但是它返回的

  2. Result 对象还保留着连接的引用,当后面的代码需要处理结果集的时候,连接将会被重新启用 DB.Query() 调用完毕后将连接传递给

  3. sql.Rows 类型,当后者迭代完毕或者显示的调用 Close() 方法后,连接将会被释放到连接池 DB.QueryRow()

  4. 调用完毕后将连接传递给 sql.Row 类型,当 Scan() 方法调用完成后,连接将会被释放到连接池 DB.Begin()调用完毕后将连接传递给 sql.Tx 类型对象,当 Commit() 或 Rollback() 方法调用后释放连接

每个连接都是惰性的,如果验证 sqlx.Open() 调用之后,sqlx.DB 类型对象可用呢?通过 DB.Ping() 方法来初始化

参考文章:https://www.cnblogs.com/kaichenkai/p/11140555.html

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值