Go语言学习之操作MYSQL实现CRUD

目录
  • 1.介绍
  • 2.下载安装驱动
    • 2.1 安装驱动
  • 3.匿名导入
  • 4.连接数据库
    • 4.1 连接方法
    • 4.2 sql.DB作用
    • 4.3 sql.DB设计目标
  • 5.写操作(增、删、改)
    • 5.1 执行步骤
    • 5.2 代码示例
  • 6. 读操作(查询)
    • 6.1 执行步骤
    • 6.2 代码示例
    • 6.3 注意事项
    • 6.4 为什么查询后要关闭连接

1.介绍

Go官方提供了database包,database包下有sql/driver。该包用来定义操作数据库的接口,这保证了无论使用哪种数据库,操作方式都是相同的。但Go官方并没有提供连接数据库的driver,如果要操作数据库,还需要第三方的driver包。

2.下载安装驱动

go-sql-driver驱动源码地址: https://github.com/go-sql-driver/mysql

2.1 安装驱动

go get -u github.com/go-sql-driver/mysql

3.匿名导入

通常来说,导入包后就能调用该包中的数据和方法。但是对于数据库操作来说,不应该直接使用导入驱动包所提供的方法,而应该使用 sql.DB对象所提供的统一的方法。因此在导入MySQL驱动时,使用了匿名导入包的方式。

匿名导入包: 只导入包但是不使用包内的类型和数据,使用匿名的方式: 在包路径前添加下画线_

import (
    _ "github.com/go-sql-driver/mysql"
)

在导入一个数据库驱动后,该驱动会自行初始化并注册到Godatabase/sql上下文中,这样就可以通过 database/sql 包所提供的方法来访问数据库了。

4.连接数据库

4.1 连接方法

使用sql包中的Open()函数来连接数据库。

Open(driverName, dataSourceName string) (*DB, error)
  • driverName: 使用的驱动名,如mysql。(注册到 database/sql时所使用的名字)
  • dataSourceName:数据库连接信息,格式:[用户名:密码@tcp(IP:port)/数据库?charset=utf8],例如:root:123456@tcp(127.0.0.1:3306)/test?charset=utf8

4.2 sql.DB作用

  • sql.Open()返回的sql.DB对象是Goroutine并发安全的。
  • sql.DB 通过数据库驱动为开发者提供管理底层数据库连接的打开和关闭操作。
  • sql.DB 帮助开发者管理数据库连接池。正在使用的连接被标记为繁忙,用完后回到连接池等待下次使用。所以,<font color=red>如果开发者没有把连接释放回连接池,会导致过多连接使系统资源耗尽。</font>

4.3 sql.DB设计目标

sql.DB的设计目标就是作为长连接(一次连接多次数据交互)使用,不宜频繁开关。比较好的做法是,为每个不同的datastore建一个DB对象,保持这些对象打开。如果需要短连接(一次连接一次数据交互),就把DB作为参数传入function,而不要在function中开关。

5.写操作(增、删、改)

5.1 执行步骤

  • 先使用预编译语句(PreparedStatement)来拼接sql
  • 然后调用db.Exec()执行SQL,返回执行结果

5.2 代码示例

package main

import (
    "database/sql"
    "fmt"
    _ "github.com/go-sql-driver/mysql"
    "time"
)
func main() {
    // 连接数据库
    open, err := sql.Open("mysql", "root:root@tcp(127.0.0.1:3306)/test?charset=utf8")
    checkError(err)
  //插入数据
    //add(open)
    // 更新数据
    //update(open)
    // 删除数据
    del(open)
}
//插入数据
func add(open *sql.DB)  {
    //插入数据
    prepare, err := open.Prepare("insert user set username=?,password=?,mobile=?,createtime=?")
    checkError(err)
    exec, err := prepare.Exec("李四", "123456", "17600000000", time.Now().Unix())
    checkError(err)
    id, err := exec.LastInsertId()
    checkError(err)
    fmt.Printf("插入数据ID: %d \n",id)
}
// 更新
func update(open *sql.DB)  {
    prepare, err := open.Prepare("update user set username=? where id =?")
    checkError(err)
    exec, err := prepare.Exec("王五", "18")
    checkError(err)
    rows, err := exec.RowsAffected()
    checkError(err)
    fmt.Printf("更新数据成功,影响条数 %d \n",rows)
}
// 删除数据
func del(open *sql.DB)  {
    prepare, err := open.Prepare("delete from user  where id =?")
    checkError(err)
    exec, err := prepare.Exec( "8")
    checkError(err)
    rows, err := exec.RowsAffected()
    checkError(err)
    fmt.Printf("删除数据成功,影响条数 %d \n",rows)
}
//检测错误
func checkError(err error)  {
    if err != nil {
        panic("操作失败:"+err.Error())
    }
}

6. 读操作(查询)

6.1 执行步骤

1. 查询多条步骤

  • 调用db.Query()方法执行SQL语句,返回一个Rows查询结果。
  • rows.Next()方法的返回值作为for循环的条件,迭代查询数据。
  • 在循环中,通过 rows.Scan()方法读取每一行数据。
  • 调用db.Close()关闭查询。

2.查询单条步骤

  • 调用db.QueryRow()方法执行SQL语句,返回一个Row查询结果。
  • 然后调用row.Scan()读取数据。

6.2 代码示例

package main
import (
    "database/sql"
    "fmt"
    _ "github.com/go-sql-driver/mysql"
)
func main() {
    // 连接数据库
    db, err := sql.Open("mysql", "root:root@tcp(127.0.0.1:3306)/nsbd_app?charset=utf8")
    checkError(err)
    //查询多条数据
    rows := queryRows(db)
    fmt.Printf("多条返回: \n%+v\n",rows)
    // 查询单条数据
    row := queryRow(db)
    fmt.Printf("单条返回: \n%+v\n",row)
}

// 创建表的映射对象
type User struct {
    Uid        int
    UserName   string
    CreateTime int
    Birthday   sql.NullString //有的值可能为NULL
}

//查询多条数据
func queryRows(db *sql.DB) []User {
    stmt, err := db.Prepare("select id,username,createtime,birthday from nsbd_user where id < ?")
    checkError(err)
    rows, err := stmt.Query(30)
    // 延迟关闭
    defer rows.Close()
    checkError(err)
    user := new(User)
    //users := make([]User,5)
    var users []User
    for rows.Next() {
        // rows.Scan()方法的参数顺序很重要,必须和查询结果的column相对应(数量和顺序都需要一致)
        err := rows.Scan(&user.Uid, &user.UserName, &user.CreateTime, &user.Birthday)
        checkError(err)
        users = append(users, *user)
    }
    return users
}
// 查询单条数据
func queryRow(db *sql.DB) User {
    stmt, err := db.Prepare("select id,username,createtime,birthday from nsbd_user where id = ?")
    checkError(err)
    user := new(User)
    err = stmt.QueryRow(4).Scan(&user.Uid, &user.UserName, &user.CreateTime, &user.Birthday)
    checkError(err)
    return *user
}
//检测错误
func checkError(err error) {
    if err != nil {
        panic("操作失败:" + err.Error())
    }
}

输出:

多条返回: 
[{Uid:1 UserName:admin CreateTime:0 Birthday:{String:2017-04-15 Valid:true}} {Uid:2 UserName:u2 CreateTime:1605858072 Birthday:{String:1993-02-14 Valid:true}} {Uid:3 UserName:u3 CreateTime:1606289644 Birthday:{String:1991-05-31 Valid:true}} {Uid:4 UserName:u4 CreateTime:1610521164 Birthday:{String:1989-11-24 Valid:true}} {Uid:5 UserName:u5 CreateTime:1610588359 Birthday:{String: Valid:false}}]
单条返回: 
{Uid:4 UserName:u4 CreateTime:1610521164 Birthday:{String:1989-11-24 Valid:true}}

6.3 注意事项

  • rows.Scan()方法的参数顺序很重要,必须和查询结果的column相对应(数量和顺序都需要一致);
  • Go是强类型语言,在查询数据时先定义数据类型,针对字段值为NULL时,数据类型应定义为:sql.NullString、sql.NullInt64等,并可以通过Valid值来判断查询到的值是赋值状态还是未赋值状态。
  • 每次db.Query()操作后,都建议调用rows.Close()rows.Close()操作是幂等操作,即便对已关闭的rows再执行close()也没关系。

6.4 为什么查询后要关闭连接

因为db.Query()会从数据库连接池中获取一个连接,这个底层连接在结果集(rows)未关闭前会被标记为处于繁忙状态。当遍历读到最后一条记录时,会发生一个内部EOF错误,自动调用rows.Close()。但如果出现异常,提前退出循环,rows不会关闭,连接不会回到连接池中,连接也不会关闭,则此连接会一直被占用。因此通常使用defer rows.Close()来确保数据库连接可以正确放回到连接池中。

以上就是Go语言学习之操作MYSQL实现CRUD的详细内容,更多关于Go语言操作MYSQL实现CRUD的资料请关注我们其它相关文章!

(0)

相关推荐

  • golang连接mysql数据库操作使用示例

    目录 安装 连接数据库 处理类型(Handle Types) 建表 Exec使用 Exec增删该示例 sql预声明(Prepared Statements) Query Queryx QueryRow和QueryRowx Get 和Select(非常常用) 事务(Transactions) 连接池设置 案例使用 golang操作mysql 安装 go get "github.com/go-sql-driver/mysql" go get "github.com/jmoiron

  • Mysql数据类型与CRUD操作详细讲解

    目录 基本数据类型 数据库命令 建表与约束 建表 约束 基本数据操作(CRUD) 基本数据类型 整数:可选择unsigned修饰 intyint 8位 (-128 - 127) smallint 16位 (-32768 - 32767) mediumint 24位 (-8388608 - 8388607) int 32位 大约正负21亿 bigint 64位 实数(带有小数点):使用标准的浮点运算进行近似计算 float 4个字节 double 8个字节 decimal 最多允许65个数字 示例

  • Go单元测试对数据库CRUD进行Mock测试

    目录 go-sqlmock 安装 使用示例 miniredis 安装 使用示例 总结 最近在实践中也总结了一些如何用表格驱动的方式使用 gock Mock测试外部接口调用.以及怎么对GORM做mock测试,这些等这篇学完基础后,后面再单独写文章给大家介绍. 这是Go语言单元测试系列教程的第3篇,介绍了如何使用go-sqlmock和miniredis工具进行MySQL和Redis的mock测试. 在上一篇<Go单元测试--模拟服务请求和接口返回>中,我们介绍了如何使用httptest和gock工

  • MySQL 详细单表增删改查crud语句

    MySQL 增删改查语句 1.创建练习表 这里练习表没有满足三范式 第一范式(又称 1NF):保证每列的原子性 数据表中的每一列(字段),必须是不可拆分的最小单元,也就是确保每一列的原子性.满足第一范式是关系模式规范化的最低要求,否则,将有很多基本操作在这样的关系模式中实现不了. 第二范式(又称 2NF):保证一张表只描述一件事情 满足1NF后要求表中的所有列,每一行的数据只能与其中一列相关,即一行数据只做一件事.只要数据列中出现数据重复,就要把表拆分开来. 第三范式(又称 3NF):保证每列都

  • Go语言实现操作MySQL的基础知识总结

    目录 前言 下载依赖 使用 MySQL 驱动 SetMaxOpenConns SetMaxIdleConns 操作 前言 MySQL 是目前开发中最常见的关系型数据库,也是程序员打交道最多的数据库. 使用 Go 语言进行操控数据库需要使用 Go 自带database/sql和驱动go-sql-driver/mysql来实现.这篇文章主要总结一下怎么使用Go语言操作MySql数据库,需要的朋友可以参考以下内容,希望对大家有帮助. 下载依赖 database/sql是Go操作数据库的标准库之一,它提

  • Go语言学习之操作MYSQL实现CRUD

    目录 1.介绍 2.下载安装驱动 2.1 安装驱动 3.匿名导入 4.连接数据库 4.1 连接方法 4.2 sql.DB作用 4.3 sql.DB设计目标 5.写操作(增.删.改) 5.1 执行步骤 5.2 代码示例 6. 读操作(查询) 6.1 执行步骤 6.2 代码示例 6.3 注意事项 6.4 为什么查询后要关闭连接 1.介绍 Go官方提供了database包,database包下有sql/driver.该包用来定义操作数据库的接口,这保证了无论使用哪种数据库,操作方式都是相同的.但Go官

  • 用C语言操作MySQL数据库的通用方法

    在我们的web应用中,虽然PHP.JSP等脚本均提供了MySQL的接口,但是显然直接使用C语言具有更好的安全性和性能,在这篇文章中能够有所体现. 先看结构体: 以下代码块是用来连接数据库的通讯过程,要连接MYSQL,必须建立MYSQL实例,通过mysql_init初始化方能开始进行连接. typedef struct st_mysql { NET net; /* Communication parameters */ gptr connector_fd; /* ConnectorFd for S

  • Go语言操作MySQL的知识总结

    目录 一.连接 1.1 下载依赖 1.2 使用MySQL驱动 1.3 初始化连接 1.4 SetMaxOpenConns 1.5 SetMaxIdleConns 二.CRUD 2.1 建库建表 2.2 查询 2.3 单行查询 2.4 多行查询 2.5 插入数据 2.6 更新数据 2.7 删除数据 三.事务 3.1 什么是事务 3.2 事务的ACID 3.3 事务相关方法 3.4 事务示例 一.连接 Go语言中的database/sql包提供了保证SQL或类SQL数据库的泛用接口,并不提供具体的数

  • Spring mvc整合mybatis(crud+分页插件)操作mysql

    一.web.xml配置 我们都知道java ee的项目启动的第一件事就是读取web.xml,spring mvc 的web.xml我在上一篇文章中也做了详细讲解,不懂的可以回头看看,讲解的这个项目源码我也会放到github上,也可以去那里看看,这里就不做介绍了. web.xml 配置 <context-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:/c

  • R语言实现操作MySQL数据库

    用R语言做数据分析时,常常需要从多种数据源取数据,其中数据库是非常常见的数据源.用R操作MySQL数据库,可以说是数据分析师必备的技能了,本文介绍RMySQL包,可以在R语言中对数据库进行增删改查的操作. 软件版本 win10 64bit r3.6.1 rstudio 1.2 RMySQL 0.10.20 安装包 install.packages('RMySQL') 创建连接 用dbConnect函数创建连接,驱动类型设置为MySQL(),用户名user.密码password.主机host.端口

  • Go语言学习笔记之文件读写操作详解

    目录 文件写 文件读 小结 文件操作比较多,分为几篇来写吧.首先是文件的读写,在平时的工程化操作中使用最多. 文件写 样例代码如下 package main import ( "bufio" "fmt" "io" "os" ) //写文件 func DoWriteFile() error { _filePath := "./test.txt" _file, _err := os.OpenFile(_file

  • Python操作MySQL数据库9个实用实例

    在Windows平台上安装mysql模块用于Python开发 用python连接mysql的时候,需要用的安装版本,源码版本容易有错误提示.下边是打包了32与64版本. MySQL-python-1.2.3.win32-py2.7.exe MySQL-python-1.2.3.win-amd64-py2.7.exe 实例 1.取得 MYSQL 的版本 # -*- coding: UTF-8 -*- #安装 MYSQL DB for python import MySQLdb as mdb con

  • 在Python程序中操作MySQL的基本方法

    Python操作Mysql 最近在学习python,这种脚本语言毫无疑问的会跟数据库产生关联,因此这里介绍一下如何使用python操作mysql数据库.我python也是零基础学起,所以本篇博客针对的是python初学者,大牛可以选择绕道. 另外,本篇基于的环境是Ubuntu13.10,使用的python版本是2.7.5. MYSQL数据库 MYSQL是一个全球领先的开源数据库管理系统.它是一个支持多用户.多线程的数据库管理系统,与Apache.PHP.Linux共同组成LAMP平台,在web应

  • Python操作MySQL模拟银行转账

    今天在慕课网上学习了有关于python操作MySQL的相关知识,在此做些总结.python操作数据库还是相对比较简单的,由于python统一了各个数据库的接口程序,也就是所谓的Python DB,所以无论使用何种数据可,都可以用统一的接口对数据库进行操作.操作中主要涉及connection对象的操作和cursor的操作,前者主要是为了建立起python与数据库的数据交换通道,后者则是访问数据的游标,也可以理解为指针.数据库的相关结构化语言在Python中均是以字符串的形式呈现的.另外注意roll

随机推荐