龿¥æ°æ®åº
声æå ¨å±åé + å®ä¹ç»æä½ + è¿æ¥æ°æ®åº
// 声æå
¨å±åédb
var db *sqlx.DB
// ç»æä½åå¨user表å
è¾åº
type User struct {
ID int
Age int `db:"age"`
Name string `db:"name"`
}
// è¿æ¥æ°æ®åº
func initDB() (err error) {
dsn := "root:wang[email protected](127.0.0.1:3306)/db1?charset=utf8mb4&parseTime=True"
// è¿æ¥æ°æ®åº open + ping
// db = sqlx.MustConnect("mysql", dsn) // è¿æ¥ä¸æåç´æ¥panic
db, err = sqlx.Connect("mysql", dsn)
if err != nil {
fmt.Printf("connect DB failed, err:%v\n", err)
return
}
// æå¤§è¿æ¥æ°
db.SetMaxOpenConns(20)
// 空é²é¾æ¥æ°
db.SetMaxIdleConns(10)
return
}
æ¥è¯¢
- åæ¡æ°æ®æ¥è¯¢ /
db.Get()
- 夿¡æ°æ®æ¥è¯¢ /
db.Select()
- ç»æä½ä¸å段ç»å®æ¥è¯¢ /
db.NamedQuery()
- å¤åæ°æ¥è¯¢ /
sqlx.In()
// 1. æ¥è¯¢åæ¡æ°æ® / db.Get() / ç¸å½äº query + scan
func queryRowDemo() {
// sqlè¯å¥ ââ> å符串
sqlStr := "select id, name, age from user where id=?"
var u User
// æå®id为1 / å°æ°æ®åå
¥ç»æä½å®ä¾ä¸
err := db.Get(&u, sqlStr, 1)
if err != nil {
fmt.Printf("get failed, err:%v\n", err)
return
}
fmt.Printf("id:%d name:%s age:%d\n", u.ID, u.Name, u.Age)
}
// 2. æ¥è¯¢å¤æ¡æ°æ® / db.Select()
func queryMultiRowDemo() {
// sqlè¯å¥ ââ> å符串
sqlStr := "select id, name, age from user where id > ?"
var users []User
// æå®id>1 / å°æ°æ®åå
¥ç»æä½å®ä¾ä¸
err := db.Select(&users, sqlStr, 1)
if err != nil {
fmt.Printf("query failed, err:%v\n", err)
return
}
fmt.Printf("users:%#v\n", users)
}
// 3. ç»å®SQLè¯å¥ä¸ç»æä½æmapä¸çåååæ®µï¼è¿è¡æ¥è¯¢æä½ / db.NamedQuery() + rows.Next()
func namedQuery(){
sqlStr := "SELECT * FROM user WHERE name=:name"
// 1. 使ç¨mapåå½åæ¥è¯¢
rows, err := db.NamedQuery(sqlStr,
map[string]interface{}{"name": "Tom"},
)
if err != nil {
fmt.Printf("db.NamedQuery failed, err:%v\n", err)
return
}
defer rows.Close() // å»¶æ¶å
³é
// 循ç¯åå
¥userç»æä½å®ä¾ï¼å¹¶è¾åº
for rows.Next(){
var u User
err := rows.StructScan(&u)
if err != nil {
fmt.Printf("scan failed, err:%v\n", err)
continue
}
fmt.Printf("user:%#v\n", u)
}
// 4. æ¹éæ¥è¯¢ æ ¹æ®æ¹éIDæ¥è¯¢æ°æ® / sqlx.In() + db.Rebind() + db.Select()
/*
users,_ := QueryByIDs([]int{4,5,6})
for _, user := range users {
fmt.Printf("user:%#v\n", user)
}
*/
func QueryByIDs(ids []int)(users []User, err error){
// 卿填å
id
query, args, err := sqlx.In("SELECT name, age FROM user WHERE id IN (?)", ids)
if err != nil {
return
}
// sqlx.In è¿å带 `?` bindvarçæ¥è¯¢è¯å¥, Rebind()å¯ä»¥éæ°ç»å®å®
query = db.Rebind(query)
err = db.Select(&users, query, args...)
return
}
æå ¥
- æå
¥åæ¡ï¼è¡ï¼æ°æ® /
db.Exec()
- ç»æä½ä¸å段ç»å®æå
¥ /
db.NamedExec()
- æ¹éæå
¥æ°æ® /
sqlx.In()
// 1. æå
¥åæ¡æ°æ® / db.Exec()
func insertRowDemo() {
sqlStr := "insert into user(name, age) values (?,?)"
ret, err := db.Exec(sqlStr, "ä¾ å¥¢", 18)
if err != nil {
fmt.Printf("insert failed, err:%v\n", err)
return
}
// è·åæ°æå
¥æ°æ®çid / éªè¯æ¯å¦æå
¥æå
theID, err := ret.LastInsertId()
if err != nil {
fmt.Printf("get lastinsert ID failed, err:%v\n", err)
return
}
fmt.Printf("insert success, the id is %d.\n", theID)
}
// 2. ç»å®SQLè¯å¥ä¸ç»æä½æmapä¸çåååæ®µï¼è¿è¡æå
¥æä½ / db.NamedExec()
func insertUserDemo()(err error){
sqlStr := "INSERT INTO user (name,age) VALUES (:name,:age)"
_, err = db.NamedExec(sqlStr,
map[string]interface{}{
"name": "Tom",
"age": 13,
})
return
}
// 3. æ¹éæå
¥æ°æ® sqlx.In / 注æä¼ å
¥çåæ°æ¯[]interface{}
// 3.1 ç»æä½å®ç°driver.Valueræ¥å£
func (u User) Value() (driver.Value, error) {
return []interface{}{u.Name, u.Age}, nil
}
// 3.2 sqlx.Inæ¼æ¥ SQL è¯å¥å æå
¥åæ° , 注æä¼ å
¥çåæ°æ¯[]interface{}
/*
u1 := User{Age: 1, Name: "Daming"}
u2 := User{Age: 2, Name: "Xiaoming"}
u3 := User{Age: 3, Name: "Zhongming"}
users := []interface{}{u1, u2, u3}
BatchInsertUsers2(users)
*/
func BatchInsertUsers2(users []interface{}) error {
query, args, _ := sqlx.In(
"INSERT INTO user (name, age) VALUES (?), (?), (?)",
users..., // 妿argå®ç°äº driver.Valuer, sqlx.In ä¼éè¿è°ç¨ Value()æ¥å±å¼å®
)
fmt.Println(query) // æ¥ççæçquerystring
fmt.Println(args) // æ¥ççæçargs
_, err := db.Exec(query, args...)
return err
}
æ´æ°
// æ´æ°æ°æ® / db.Exec()
func updateRowDemo() {
sqlStr := "update user set age=? where id = ?"
ret, err := db.Exec(sqlStr, 39, 6)
if err != nil {
fmt.Printf("update failed, err:%v\n", err)
return
}
// æä½å½±åçè¡æ°
n, err := ret.RowsAffected()
if err != nil {
fmt.Printf("get RowsAffected failed, err:%v\n", err)
return
}
fmt.Printf("update success, affected rows:%d\n", n)
}
å é¤
// å 餿°æ® / db.Exec()
func deleteRowDemo() {
sqlStr := "delete from user where id = ?"
ret, err := db.Exec(sqlStr, 6)
if err != nil {
fmt.Printf("delete failed, err:%v\n", err)
return
}
// æä½å½±åçè¡æ°
n, err := ret.RowsAffected()
if err != nil {
fmt.Printf("get RowsAffected failed, err:%v\n", err)
return
}
fmt.Printf("delete success, affected rows:%d\n", n)
}
äºå¡
// äºå¡æä½ / db.Beginx()
func transactionDemo2()(err error) {
// å¼å¯äºå¡
tx, err := db.Beginx()
if err != nil {
fmt.Printf("begin trans failed, err:%v\n", err)
return err
}
defer func() {
// æè·panic
if p := recover(); p != nil {
// åæ»
tx.Rollback()
panic(p) // re-throw panic after Rollback
} else if err != nil {
fmt.Println("rollback")
tx.Rollback() // err is non-nil; don't change it
} else {
err = tx.Commit() // err is nil; if Commit returns error update err
fmt.Println("commit")
}
}()
// æ´æ°è¯å¥1
sqlStr1 := "Update user set age=18 where id=?"
rs, err := tx.Exec(sqlStr1, 1)
if err!= nil{
return err
}
n, err := rs.RowsAffected() // å½±åè¡æ°
if err != nil {
return err
}
if n != 1 {
return errors.New("exec sqlStr1 failed")
}
// æ´æ°è¯å¥2
sqlStr2 := "Update user set age=99 where i=?"
rs, err = tx.Exec(sqlStr2, 5)
if err!=nil{
return err
}
n, err = rs.RowsAffected() // å½±åè¡æ°
if err != nil {
return err
}
if n != 1 {
return errors.New("exec sqlStr1 failed")
}
return err
}