mysqlsqlx
Posted yihengye
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了mysqlsqlx相关的知识,希望对你有一定的参考价值。
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](https://zhuanlan.zhihu.com/p/430993402)
总结:
只用 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() 方法来初始化
连接池配置
DB.SetMaxIdleConns(n int) 设置连接池中的空闲连接的最大连接数。默认也是0,表示连接池不会保持数据库连接的状态:即当连接释放回到连接池的时候,连接将会被关闭。这会导致连接再连接池中频繁的关闭和创建,我们可以设置一个合理的值。
DB.SetMaxOpenConns(n int) 设置打开数据库的最大连接数。包含正在使用的连接和连接池的连接。如果你的方法调用 需要用到一个连接,并且连接池已经没有了连接或者连接数达到了最大连接数。此时的方法调用将会被 block,直到有可用的连接才会返回。设置这个值可以避免并发太高导致连接 mysql 出现 too many connections 的错误。该函数的默认设置是0,表示无限制
DB.SetConnMaxLifetime(d time.Duration) 设置连接可以被使用的最长有效时间,如果过期,连接将被拒绝
参考文章:https://www.cnblogs.com/kaichenkai/p/11140555.html
推荐适合初级程序员的AI智能搜索网站
Phind自称是为开发人员而生的AI搜索引擎(The AI search engine for developers)。
与ChatGPT和new Bing一样,Phind由大语言模型(Large Language Model (LLM))驱动。体验后,个人感觉在技术方面的检索能力和质量上Phind 比 new Bing 和 ChatGPT 的体验要好得多。
Phind也支持非开发人员相关问题回答,响应速度和质量也不错,关键不需要new Bing的waitlist
以上是关于mysqlsqlx的主要内容,如果未能解决你的问题,请参考以下文章