Go语言中使用 sqlx 来操作 MySQL

本文涉及的产品
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 高可用系列,价值2615元额度,1个月
简介: Go语言因其高效的性能和简洁的语法而受到开发者们的欢迎。在开发过程中,数据库操作不可或缺。虽然Go的标准库提供了`database/sql`包支持数据库操作,但使用起来稍显复杂。为此,`sqlx`应运而生,作为`database/sql`的扩展库,它简化了许多常见的数据库任务。本文介绍如何使用`sqlx`包操作MySQL数据库,包括安装所需的包、连接数据库、创建表、插入/查询/更新/删除数据等操作,并展示了如何利用命名参数来进一步简化代码。通过`sqlx`,开发者可以更加高效且简洁地完成数据库交互任务。

Go 语言以其高效和简洁的语法逐渐受到开发者的青睐。在实际开发中,数据库操作是不可避免的任务之一。虽然标准库提供了 database/sql 包来支持数据库操作,但使用起来略显繁琐。

sqlx 包作为一个扩展库,它在 database/sql 的基础上,提供了更高级别的便利,极大地简化了数据库操作。本文章将介绍如何通过 github.com/jmoiron/sqlx 包来操作 MySQL 数据库。

准备工作

首先,确保你的 Go 环境已经搭建完毕,并且 MySQL 数据库已安装并正在运行。接下来,安装 sqlx 包及 MySQL 驱动:

go get github.com/jmoiron/sqlx
go get github.com/go-sql-driver/mysql
AI 代码解读

连接 MySQL 数据库

在使用数据库之前,我们需要建立与 MySQL 的连接。在 Go 语言中,通常使用一个连接字符串来指定数据库的一些信息。以下是一个示例代码,演示如何连接 MySQL 数据库:

package main

import (
    "fmt"

    _ "github.com/go-sql-driver/mysql" // 一定不能忘记导入数据库驱动
    "github.com/jmoiron/sqlx"
)

var db *sqlx.DB

type User struct {
   
    ID   int64  `db:"id"`
    Name string `db:"name"`
    Age  int    `db:"age"`
}

func initDB() (err error) {
   
    dsn := "root:123456@tcp(127.0.0.1:3306)/sql_test?charset=utf8mb4&parseTime=True"
    // 也可以使用 MustConnect 连接不成功就直接 panic
    // db = sqlx.MustConnect("mysql", dsn)
    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
}
AI 代码解读

在这个例子中,请替换为你自己的MySQL 配置。

数据库操作

1. 创建表

接下来,让我们创建一个示例表。我们可以使用 Exec 方法执行 SQL 语句来创建表。

func CreateTable(db *sqlx.DB) (err error) {
   
    // 写SQL语句
    sqlStr := `create table if not exists users (
        id bigint primary key auto_increment,
        name varchar(20),
        age int default 1
    );`
    _, err = db.Exec(sqlStr)

    return err
}
AI 代码解读

main 函数中调用 CreateTable(db),以确保在连接后创建表。

2. 插入数据

// 插入用户并获取 ID
func insertUser(db *sqlx.DB, name string, age int) (int64, error) {
   
    result, err := db.Exec("INSERT INTO users(name, age) VALUES(?, ?)", name, age)
    if err != nil {
   
        return 0, err
    }

    id, err := result.LastInsertId()
    if err != nil {
   
        return 0, err
    }
    return id, nil
}
AI 代码解读

3. 查询数据

// 查询单条用户记录
func getUser(db *sqlx.DB, id int64) (*User, error) {
   
    var user User
    err := db.Get(&user, "SELECT * FROM users WHERE id=?", id)
    if err != nil {
   
        return nil, err
    }
    return &user, nil
}

// 查询所有用户记录
func getAllUsers(db *sqlx.DB, id int64) ([]User, error) {
   
    var users []User
    err := db.Select(&users, "SELECT * FROM users where id > ?", id)
    if err != nil {
   
        return nil, err
    }
    return users, nil
}
AI 代码解读

4. 更新数据

// 更新用户信息
func updateUser(db *sqlx.DB, id int64, name string, age int) (int64, error) {
   
    result, err := db.Exec("UPDATE users SET name=?, age=? WHERE id=?", name, age, id)
    if err != nil {
   
        return 0, err
    }
    rowsAffected, err := result.RowsAffected()
    if err != nil {
   
        return 0, err
    }
    return rowsAffected, nil
}
AI 代码解读

5. 删除数据

// 删除用户记录
func deleteUser(db *sqlx.DB, id int64) (int64, error) {
   
    result, err := db.Exec("DELETE FROM users WHERE id=?", id)
    if err != nil {
   
        return 0, err
    }
    rowsAffected, err := result.RowsAffected()
    if err != nil {
   
        return 0, err
    }
    return rowsAffected, nil
}
AI 代码解读

6. 使用命名参数来操作

// 使用命名参数插入用户
func insertUserNamed(db *sqlx.DB, name string, age int) (int64, error) {
   
    query := `INSERT INTO users(name, age) VALUES(:name, :age)`
    result, err := db.NamedExec(query, map[string]interface{
   }{
   
        "name": name,
        "age":  age,
    })
    if err != nil {
   
        return 0, err
    }
    id, err := result.LastInsertId()
    if err != nil {
   
        return 0, err
    }
    return id, nil
}

// 使用命名参数查询用户
func getUsersNamed(db *sqlx.DB, name string) ([]User, error) {
   
    query := `SELECT * FROM users WHERE name = :name`
    var users []User
    rows, err := db.NamedQuery(query, map[string]interface{
   }{
   
        "name": name,
    })
    if err != nil {
   
        return nil, err
    }
    defer rows.Close()
    for rows.Next() {
   
        var user User
        err := rows.StructScan(&user)
        if err != nil {
   
            fmt.Printf("scan failed, err:%v\n", err)
            continue
        }
        users = append(users, user)
    }

    return users, nil
}
AI 代码解读

7. 测试一下代码

func Run() {
   
    // 初始化数据库
    err := initDB()
    if err != nil {
   
        fmt.Printf("init DB failed, err:%v\n", err)
        return
    }
    defer db.Close() // 注意这行代码要写在上面err判断的下面

    // 创建表
    err = CreateTable(db)
    if err != nil {
   
        fmt.Printf("create table failed, err:%v\n", err)
        return
    }

    // 插入数据
    id, err := insertUser(db, "Alex", 18)
    if err != nil {
   
        fmt.Printf("insert user failed, err:%v\n", err)
        return
    }
    fmt.Println("insert success, the id is:", id)

    // 查询单条数据
    user, err := getUser(db, id)
    if err != nil {
   
        fmt.Printf("get user failed, err:%v\n", err)
        return
    }

    fmt.Printf("user:%#v\n", user)

    // 查询多条数据
    users, err := getAllUsers(db, 0)
    if err != nil {
   
        fmt.Printf("get all users failed, err:%v\n", err)
        return
    }

    fmt.Printf("users:%#v\n", users)

    // 更新数据
    rowsAffected, err := updateUser(db, id, "Alex", 20)
    if err != nil {
   
        fmt.Printf("update user failed, err:%v\n", err)
        return
    }

    fmt.Println("update success, affected rows:", rowsAffected)

    // 删除数据
    rowsAffected, err = deleteUser(db, id)
    if err != nil {
   
        fmt.Printf("delete user failed, err:%v\n", err)
        return
    }

    fmt.Println("delete success, affected rows:", rowsAffected)

    // 使用命名参数插入数据
    id, err = insertUserNamed(db, "Alex", 19)
    if err != nil {
   
        fmt.Printf("insert user named failed, err:%v\n", err)
        return
    }

    fmt.Println("insert named success, the id is:", id)

    // 使用命名参数查询数据
    users, err = getUsersNamed(db, "Alex")
    if err != nil {
   
        fmt.Printf("get users named failed, err:%v\n", err)
        return
    }

    fmt.Printf("users named:%#v\n", users)

    fmt.Println("exec SQL success")
}
AI 代码解读

我们可以看到,使用 sqlx 还是要比 database/sql 要简洁许多。

总结

通过 sqlx 包,我们可以更简单地在 Go 中与 MySQL 数据库进行交互,减少了样板代码并提高了代码的可读性。

希望这篇文章能帮助你更好地理解如何在 Go 中使用 sqlx 操作 MySQL 数据库!

相关实践学习
如何快速连接云数据库RDS MySQL
本场景介绍如何通过阿里云数据管理服务DMS快速连接云数据库RDS MySQL,然后进行数据表的CRUD操作。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助     相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
目录
打赏
0
1
1
0
116
分享
相关文章
监控局域网其他电脑:Go 语言迪杰斯特拉算法的高效应用
在信息化时代,监控局域网成为网络管理与安全防护的关键需求。本文探讨了迪杰斯特拉(Dijkstra)算法在监控局域网中的应用,通过计算最短路径优化数据传输和故障检测。文中提供了使用Go语言实现的代码例程,展示了如何高效地进行网络监控,确保局域网的稳定运行和数据安全。迪杰斯特拉算法能减少传输延迟和带宽消耗,及时发现并处理网络故障,适用于复杂网络环境下的管理和维护。
揭秘 Go 语言中空结构体的强大用法
Go 语言中的空结构体 `struct{}` 不包含任何字段,不占用内存空间。它在实际编程中有多种典型用法:1) 结合 map 实现集合(set)类型;2) 与 channel 搭配用于信号通知;3) 申请超大容量的 Slice 和 Array 以节省内存;4) 作为接口实现时明确表示不关注值。此外,需要注意的是,空结构体作为字段时可能会因内存对齐原因占用额外空间。建议将空结构体放在外层结构体的第一个字段以优化内存使用。
|
1月前
|
Go 语言入门指南:切片
Golang中的切片(Slice)是基于数组的动态序列,支持变长操作。它由指针、长度和容量三部分组成,底层引用一个连续的数组片段。切片提供灵活的增减元素功能,语法形式为`[]T`,其中T为元素类型。相比固定长度的数组,切片更常用,允许动态调整大小,并且多个切片可以共享同一底层数组。通过内置的`make`函数可创建指定长度和容量的切片。需要注意的是,切片不能直接比较,只能与`nil`比较,且空切片的长度为0。
Go 语言入门指南:切片
eino — 基于go语言的大模型应用开发框架(二)
本文介绍了如何使用Eino框架实现一个基本的LLM(大语言模型)应用。Eino中的`ChatModel`接口提供了与不同大模型服务(如OpenAI、Ollama等)交互的统一方式,支持生成完整响应、流式响应和绑定工具等功能。`Generate`方法用于生成完整的模型响应,`Stream`方法以流式方式返回结果,`BindTools`方法为模型绑定工具。此外,还介绍了通过`Option`模式配置模型参数及模板功能,支持基于前端和用户自定义的角色及Prompt。目前主要聚焦于`ChatModel`的`Generate`方法,后续将继续深入学习。
288 7
|
26天前
|
企业监控软件中 Go 语言哈希表算法的应用研究与分析
在数字化时代,企业监控软件对企业的稳定运营至关重要。哈希表(散列表)作为高效的数据结构,广泛应用于企业监控中,如设备状态管理、数据分类和缓存机制。Go 语言中的 map 实现了哈希表,能快速处理海量监控数据,确保实时准确反映设备状态,提升系统性能,助力企业实现智能化管理。
34 3
eino — 基于go语言的大模型应用开发框架(一)
Eino 是一个受开源社区优秀LLM应用开发框架(如LangChain和LlamaIndex)启发的Go语言框架,强调简洁性、可扩展性和可靠性。它提供了易于复用的组件、强大的编排框架、简洁明了的API、最佳实践集合及实用的DevOps工具,支持快速构建和部署LLM应用。Eino不仅兼容多种模型库(如OpenAI、Ollama、Ark),还提供详细的官方文档和活跃的社区支持,便于开发者上手使用。
213 8
Go 语言中的 Sync.Map 详解:并发安全的 Map 实现
`sync.Map` 是 Go 语言中用于并发安全操作的 Map 实现,适用于读多写少的场景。它通过两个底层 Map(`read` 和 `dirty`)实现读写分离,提供高效的读性能。主要方法包括 `Store`、`Load`、`Delete` 等。在大量写入时性能可能下降,需谨慎选择使用场景。
Go语言实战:错误处理和panic_recover之自定义错误类型
本文深入探讨了Go语言中的错误处理和panic/recover机制,涵盖错误处理的基本概念、自定义错误类型的定义、panic和recover的工作原理及应用场景。通过具体代码示例介绍了如何定义自定义错误类型、检查和处理错误值,并使用panic和recover处理运行时错误。文章还讨论了错误处理在实际开发中的应用,如网络编程、文件操作和并发编程,并推荐了一些学习资源。最后展望了未来Go语言在错误处理方面的优化方向。
阿里双十一背后的Go语言实践:百万QPS网关的设计与实现
解析阿里核心网关如何利用Go协程池、RingBuffer、零拷贝技术支撑亿级流量。 重点分享: ① 如何用gRPC拦截器实现熔断限流; ② Sync.Map在高并发读写中的取舍。
基于 Go 语言的公司内网管理软件哈希表算法深度解析与研究
在数字化办公中,公司内网管理软件通过哈希表算法保障信息安全与高效管理。哈希表基于键值对存储和查找,如用户登录验证、设备信息管理和文件权限控制等场景,Go语言实现的哈希表能快速验证用户信息,提升管理效率,确保网络稳定运行。
28 0
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等