为了账号安全,请及时绑定邮箱和手机立即绑定

Go语言中mysql数据库操作

标签:
Go

数据的持久化是程序中必不可少的,所以编程语言中对数据库的操作是非常重要的一块,本文介绍Go语言对mysql数据库的操作。

基本操作

建立连接

db, err := sql.Open("mysql", "root:root@tcp(127.0.0.1:3306)/betting?charset=utf8")errDeal("连接数据库", err)defer db.Close()

连接参数一般有以下几种

user@unix(/path/to/socket)/dbname?charset=utf8user:password@tcp(localhost:5555)/dbname?charset=utf8user:password@/dbnameuser:password@tcp([de:ad:be:ef::ca:fe]:80)/dbname

增删改操作

// 插入数据---------------------方法1result, err := db.Exec("INSERT INTO admin_log(admin_username,log_time,log_ip,log_type,log_desc)" +" VALUES(?,now(),?,?,?)", "admin", "127.0.0.1", "登录", "登录成功")errDeal("插入数据", err)// 查看返回信息count, err := result.RowsAffected()errDeal("查看插入数据条数", err)fmt.Printf("插入数据条数:%d\n", count)id, err := result.LastInsertId()errDeal("查看最后插入数据的id", err)fmt.Printf("最后插入数据的id:%d\n", id)// 插入数据---------------------方法2,先创建一个预处理语句,再执行stmt, err := db.Prepare("INSERT INTO admin_log(admin_username,log_time,log_ip,log_type,log_desc)" +" VALUES(?,now(),?,?,?)")result1, err1 := stmt.Exec("admin", "127.0.0.1", "登录", "登录成功")errDeal("方法2插入数据", err1)fmt.Println(result1.LastInsertId())// 删除数据result2, err := db.Exec("DELETE FROM admin_log WHERE id=?", 2014)fmt.Println(result2.RowsAffected())// 更新数据result3, err := db.Exec("UPDATE admin_log SET log_ip=? WHERE id=?", "192.168.8.9", 2017)fmt.Println(result3.RowsAffected())

单条数据查询

// 先定义存储查询结果的变量var rid intvar username, time, ip, logType stringvar desc, remark, spare interface{} // 如果字段中可能出现值为nil的情况,可以将变量申明为interface{}类型err2 := db.QueryRow("SELECT id,admin_username,log_time,log_ip,log_type,log_desc,remark,spare" +    " FROM admin_log WHERE id=?", 2017).Scan(&rid, &username, &time, &ip, &logType, &desc, &remark, &spare) // 传入的是变量的指针errDeal("查询单条数据", err2)fmt.Printf("id=%d,username=%s,time=%s,ip=%s,logType=%s,desc=%s,remark=%v,spare=%v\n", rid, username, time, ip, logType, desc, remark, spare)

多条数据查询

// 查询多条数据rows, err3 := db.Query("SELECT admin_username,log_time,log_ip,log_type,log_desc" +    " FROM admin_log WHERE id=? OR id=?", 2017, 2019)errDeal("查询多条数据", err3)// 对多条数据进行遍历for rows.Next() {    err4 := rows.Scan(&username, &time, &ip, &logType, &desc)    errDeal("遍历多条数据", err4)    fmt.Printf("username=%s,time=%s,ip=%s,logType=%s,desc=%s\n", username, time, ip, logType, desc)}// 如果查询中不指定具体字段,使用*rows, err33 := db.Query("SELECT *" +    " FROM admin_log WHERE id>?", 2017)errDeal("查询多条数据", err33)// 查询所有字段名,返回string切片columes, err333 := rows.Columns()errDeal("rows.Columns()方法调用", err333)fmt.Printf("%T----%v\n", columes, columes)var scanColumes = make([]interface{}, len(columes))var values = make([]interface{}, len(columes))for index, _ := range  scanColumes {    scanColumes[index] = &values[index]}for rows.Next() {    err4 := rows.Scan(scanColumes...)    errDeal("遍历多条数据", err4)    for i, val := range values {        if strings.EqualFold(judgeType(val), "[]uint8") {            fmt.Printf("%s(%T)==%s\t", columes[i], val, val)        } else {            fmt.Printf("%s(%T)==%v\t", columes[i], val, val)        }    }    fmt.Println()}

事务

开启事务

// 开启事务,tx是从连接池中取出一个连接,在关闭之前都是使用这个连接,提交事务和回滚事务都是操作txtx, err5 := db.Begin()errDeal("开启事务", err5)_, err6 := tx.Exec("UPDATE admin_log SET log_desc=? WHERE id=?", "测试事务222", 2019)//if err6 != nil {if err6 == nil {    tx.Rollback() // 回滚}tx.Commit() // 提交

批量插入数据

// 批量数据插入tx, err7 := db.Begin()errDeal("数据批量插入,开启事务", err7)insertValues := [][]interface{}{{"admin", "127.0.0.1", "登录", "登录成功"},{"admin", "127.0.0.1", "删除", "删除数据"},{"admin", "127.0.0.1", "退出", "退出系统"}}stmt, err8 := tx.Prepare("INSERT INTO admin_log(admin_username,log_time,log_ip,log_type,log_desc) VALUES(?,now(),?,?,?)")errDeal("数据批量插入,预处理", err8)for _, val := range  insertValues {    _, err := stmt.Exec(val...)    if err != nil {        fmt.Printf("出现错误回滚,错误信息:%v", err)        tx.Rollback()    }}tx.Commit()

sqlx的使用及批量插入

xdb, err9 := sqlx.Open("mysql", "root:root@tcp(127.0.0.1:3306)/betting?charset=utf8")errDeal("sqlx连接数据库", err9)txx, err10 := xdb.Beginx()errDeal("sqlx开启事务", err10)insertValuesx := [][]interface{}{{"admin", "127.0.0.1", "登录", "登录成功X"},{"admin", "127.0.0.1", "删除", "删除数据X"},{"admin", "127.0.0.1", "退出", "退出系统X"}}stmtx, err11 := txx.Preparex("INSERT INTO admin_log(admin_username,log_time,log_ip,log_type,log_desc) VALUES(?,now(),?,?,?)")errDeal("数据批量插入,预处理", err11)for _, val := range insertValuesx {    _, err := stmtx.Exec(val...)    if err != nil {        fmt.Printf("sqlx出现错误回滚,错误信息:%v", err)        txx.Rollback()    }}txx.Commit()

完整代码

package mainimport (    "database/sql"    _"github.com/go-sql-driver/mysql" // 这里很重要,导入自己本地使用的数据库驱动,前面是下划线,否则会报错:sql: unknown driver "mysql" (forgotten import?)    "fmt"    "strings"    "github.com/jmoiron/sqlx")func main() {    // 连接数据库,用户名:密码@协议(地址:端口)/数据库?参数=参数值,常用"用户名:密码@tcp(ip:端口)/数据库名?charset=字符集"    db, err := sql.Open("mysql", "root:root@tcp(127.0.0.1:3306)/betting?charset=utf8")    errDeal("连接数据库", err)    defer db.Close()    //=========================================================================================增删改    // 插入数据---------------------方法1    result, err := db.Exec("INSERT INTO admin_log(admin_username,log_time,log_ip,log_type,log_desc)" +        " VALUES(?,now(),?,?,?)", "admin", "127.0.0.1", "登录", "登录成功")    errDeal("插入数据", err)    // 查看返回信息    count, err := result.RowsAffected()    errDeal("查看插入数据条数", err)    fmt.Printf("插入数据条数:%d\n", count)    id, err := result.LastInsertId()    errDeal("查看最后插入数据的id", err)    fmt.Printf("最后插入数据的id:%d\n", id)    // 插入数据---------------------方法2,先创建一个预处理语句,再执行    stmt, err := db.Prepare("INSERT INTO admin_log(admin_username,log_time,log_ip,log_type,log_desc)" +        " VALUES(?,now(),?,?,?)")    result1, err1 := stmt.Exec("admin", "127.0.0.1", "登录", "登录成功")    errDeal("方法2插入数据", err1)    fmt.Println(result1.LastInsertId())    // 删除数据    result2, err := db.Exec("DELETE FROM admin_log WHERE id=?", 2014)    fmt.Println(result2.RowsAffected())    // 更新数据    result3, err := db.Exec("UPDATE admin_log SET log_ip=? WHERE id=?", "192.168.8.9", 2017)    fmt.Println(result3.RowsAffected())    //=============================================================================================查询    // 查询单条数据    // 先定义存储查询结果的变量    var rid int    var username, time, ip, logType string    var desc, remark, spare interface{} // 如果字段中可能出现值为nil的情况,可以将变量申明为interface{}类型    err2 := db.QueryRow("SELECT id,admin_username,log_time,log_ip,log_type,log_desc,remark,spare" +        " FROM admin_log WHERE id=?", 2017).Scan(&rid, &username, &time, &ip, &logType, &desc, &remark, &spare) // 传入的是变量的指针    errDeal("查询单条数据", err2)    fmt.Printf("id=%d,username=%s,time=%s,ip=%s,logType=%s,desc=%s,remark=%v,spare=%v\n", rid, username, time, ip, logType, desc, remark, spare)    // 查询多条数据    rows, err3 := db.Query("SELECT admin_username,log_time,log_ip,log_type,log_desc" +        " FROM admin_log WHERE id=? OR id=?", 2017, 2019)    errDeal("查询多条数据", err3)    // 对多条数据进行遍历    for rows.Next() {        err4 := rows.Scan(&username, &time, &ip, &logType, &desc)        errDeal("遍历多条数据", err4)        fmt.Printf("username=%s,time=%s,ip=%s,logType=%s,desc=%s\n", username, time, ip, logType, desc)    }    // 如果查询中不指定具体字段,使用*    rows, err33 := db.Query("SELECT *" +        " FROM admin_log WHERE id>?", 2017)    errDeal("查询多条数据", err33)    // 查询所有字段名,返回string切片    columes, err333 := rows.Columns()    errDeal("rows.Columns()方法调用", err333)    fmt.Printf("%T----%v\n", columes, columes)    var scanColumes = make([]interface{}, len(columes))    var values = make([]interface{}, len(columes))    for index, _ := range  scanColumes {        scanColumes[index] = &values[index]    }    for rows.Next() {        err4 := rows.Scan(scanColumes...)        errDeal("遍历多条数据", err4)        for i, val := range values {            if strings.EqualFold(checkType(val), "[]uint8") {                fmt.Printf("%s(%T)==%s\t", columes[i], val, val)            } else {                fmt.Printf("%s(%T)==%v\t", columes[i], val, val)            }        }        fmt.Println()    }    // ==================================================================================================事务    // 开启事务,tx是从连接池中取出一个连接,在关闭之前都是使用这个连接,提交事务和回滚事务都是操作tx    tx, err5 := db.Begin()    errDeal("开启事务", err5)    _, err6 := tx.Exec("UPDATE admin_log SET log_desc=? WHERE id=?", "测试事务222", 2019)    //if err6 != nil {    if err6 == nil {        tx.Rollback() // 回滚    }    tx.Commit() // 提交    // 批量数据插入    tx, err7 := db.Begin()    errDeal("数据批量插入,开启事务", err7)    insertValues := [][]interface{}{{"admin", "127.0.0.1", "登录", "登录成功"},{"admin", "127.0.0.1", "删除", "删除数据"},{"admin", "127.0.0.1", "退出", "退出系统"}}    stmt, err8 := tx.Prepare("INSERT INTO admin_log(admin_username,log_time,log_ip,log_type,log_desc) VALUES(?,now(),?,?,?)")    errDeal("数据批量插入,预处理", err8)    defer stmt.Close()    // 通过循环将每条SQL的参数写到目标表缓冲区。    for _, val := range  insertValues {        _, err := stmt.Exec(val...)        if err != nil {            fmt.Printf("出现错误回滚,错误信息:%v", err)            tx.Rollback()        }    }    tx.Commit()    // 使用sqlx批量数据插入    xdb, err9 := sqlx.Open("mysql", "root:root@tcp(127.0.0.1:3306)/betting?charset=utf8")    errDeal("sqlx连接数据库", err9)    defer xdb.close()    txx, err10 := xdb.Beginx()    errDeal("sqlx开启事务", err10)    insertValuesx := [][]interface{}{{"admin", "127.0.0.1", "登录", "登录成功X"},{"admin", "127.0.0.1", "删除", "删除数据X"},{"admin", "127.0.0.1", "退出", "退出系统X"}}    stmtx, err11 := txx.Preparex("INSERT INTO admin_log(admin_username,log_time,log_ip,log_type,log_desc) VALUES(?,now(),?,?,?)")    errDeal("数据批量插入,预处理", err11)    defer stmtx.Close()    // 通过循环将每条SQL的参数写到目标表缓冲区。    for _, val := range insertValuesx {        _, err := stmtx.Exec(val...)        if err != nil {            fmt.Printf("sqlx出现错误回滚,错误信息:%v", err)            txx.Rollback()        }    }    txx.Commit()}func errDeal(info string, err error) {    if err != nil {        panic(fmt.Sprintf("%s,错误信息:%v", info, err))    }}func checkType(val interface{}) string {    switch val.(type) {    case []uint8 :        return "[]uint8"    }    return ""}

点击查看更多内容
TA 点赞

若觉得本文不错,就分享一下吧!

评论

作者其他优质文章

正在加载中
  • 推荐
  • 评论
  • 收藏
  • 共同学习,写下你的评论
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦
今天注册有机会得

100积分直接送

付费专栏免费学

大额优惠券免费领

立即参与 放弃机会
意见反馈 帮助中心 APP下载
官方微信

举报

0/150
提交
取消