Go 操作 Mysql(二)
查询数据方法回顾整理
上一篇博客中,主要是快速过了一遍 demo 代码和 DB 类型对象中方法的使用
在整理查询数据方法的时候,使用了 Query() 方法,其实 sqlx 还提供了 QueryRow() 方法,查询单行记录,以及 Queryx() 和 QueryRowx() 方法,将查询的结果保存到结构体
所以我们通过 DB 查询数据的方法一共就有三对:
- Query() 和 QueryRow() 分别返回 sql.Rows 和 sql.Row 类型
- Queryx() 和 QueryRowx() 分别返回 sql.Rows 和 sql.Row 类型,支持将查询记录保存到结构体
- Get() 和 Select() 将查询记录保存到结构体 和 结构体切片中
Query() 方法
func (db *DB) Query(query string, args ...interface{}) (*Rows, error)
使用场景:查询字段较少的情况下使用,比如 select uid, username from userinfo; 这样的语句,如果是 select * from userinfo,就使用 Get() 或 Select() 好了
Query() 返回的结果集是 sql.Rows 类型,它有一个 Next() 方法,可以迭代数据库的游标,进而获取下一条记录,结果集使用完毕之后需要调用 rows.Close() 手动关闭连接
其实通过 for 循环迭代数据的时候,当迭代到最后一行记录时,会发出一个 io.EOF(与读文件类似),引发一个错误,同时 Go 会自动调用 rows.Close() 方法释放连接,然后返回 false,此时循环结束退出
通常情况下,会正常迭代完数据然后退出循环,可是如果因为循环语句中的其它错误导致退出了循环,此时 rows.Next() 处理结果集的过程并没有完成,归属于 rows 的数据库连接不会释放回到连接池,因此十分有必要正确的处理 rows 的连接,如果没有关闭 rows 连接,将导致大量的连接并且不会被其它方法重用,就像溢出了一样,最终导致数据库无法使用(提示数据库有过多的连接)
rows.Next循环迭代的时候,因为触发了io.EOF而退出循环。为了检查是否是迭代正常退出还是异常退出,需要检查rows.Err
package main
import (
"fmt"
_ "github.com/go-sql-driver/mysql"
"github.com/jmoiron/sqlx"
) var (
userName string = "root"
password string = "seemmo"
ipAddrees string = "10.10.4.80"
port int = 3306
dbName string = "golang_db"
charset string = "utf8"
) func connectMysql() *sqlx.DB {
dsn := fmt.Sprintf("%s:%s@tcp(%s:%d)/%s?charset=%s", userName, password, ipAddrees, port, dbName, charset)
Db, err := sqlx.Open("mysql", dsn)
if err != nil {
fmt.Printf("mysql connect failed, detail is [%v]", err.Error())
}
return Db
} func queryData(Db *sqlx.DB) {
rows, err := Db.Query("select uid, username, create_time from userinfo")
if err != nil {
fmt.Printf("query data failed, error is [%v]", err.Error())
return
} for rows.Next() {
var uid int
var userName, createTime string
err := rows.Scan(&uid, &userName, &createTime)
if err != nil {
fmt.Printf("scan data failed, error is [%v]", err.Error())
return
}
fmt.Println(uid, userName, createTime)
} err = rows.Close()
if err != nil {
fmt.Println(err.Error())
}
} func main (){
var Db *sqlx.DB = connectMysql()
defer Db.Close() queryData(Db)
} //运行结果:
//1 johny 2019-07-08 10:43:21
//2 anson 2019-07-08 10:52:46
QueryRow() 方法
func (db *DB) QueryRow(query string, args ...interface{}) *Row
Query() 方法是查询多行结果集的(sqlx.Rows),QueryRow() 方法用来查询单行结果集(sqlx.Row),不需要通过 Next() 方法迭代
QueryRow() 方法的返回值与 Query() 不同,它要么返回一个 sqlx.Row 类型,要么返回一个 error 类型,如果是发生了 error,则会延迟到 Scan() 方法调用结束后返回,如果没有错误,则 Scan 正常执行,只有当查询结果为空的时候,会触发一个 sqlx.ErrNoRows 错误,你可以先调用 Scan() 方法再检查错误(也可以先检查错误再调用 Scan() 方法)
在没有过滤条件的情况下,默认返回第一条数据,不用调用 Close() 方法释放连接(因为只有一条记录)
package main
import (
"fmt"
_ "github.com/go-sql-driver/mysql"
"github.com/jmoiron/sqlx"
"time"
) var (
userName string = "root"
password string = "seemmo"
ipAddrees string = "10.10.4.80"
port int = 3306
dbName string = "golang_db"
charset string = "utf8"
) func connectMysql() *sqlx.DB {
dsn := fmt.Sprintf("%s:%s@tcp(%s:%d)/%s?charset=%s", userName, password, ipAddrees, port, dbName, charset)
Db, err := sqlx.Open("mysql", dsn)
if err != nil {
fmt.Printf("mysql connect failed, detail is [%v]", err.Error())
}
return Db
} func queryRow(Db *sqlx.DB) {
row := Db.QueryRow("select uid, username, create_time from userinfo") var uid int
var userName, createTime string
err := row.Scan(&uid, &userName, &createTime)
if err != nil {
fmt.Printf("scan data failed, error is [%v]", err.Error())
return
}
fmt.Println(uid, userName, createTime)
} func main (){
var Db *sqlx.DB = connectMysql()
defer Db.Close() queryRow(Db)
} 运行结果:
1 johny 2019-07-08 10:43:21
查询方法补充
Queryx() 和 QueryRowx(),不仅支持 Scan() 方法,同时可将数据与结构体进行转换
1)Queryx()
func (db *DB) Queryx(query string, args ...interface{}) (*Rows, error)
代码示例
func queryx(Db *sqlx.DB) {
//定义结构体保存数据
type userinfo struct {
Uid int `db:"uid"`
UserName string `db:"username"`
CreateTime string `db:"create_time"`
}
var userData userinfo
rows, err := Db.Queryx("select uid, username, create_time from userinfo")
if err != nil {
fmt.Printf("query data failed, error is [%v]", err.Error())
return
}
var userDataSlice []userinfo
for rows.Next() {
err := rows.StructScan(&userData)
if err != nil {
fmt.Printf("scan data failed, error is [%v]", err.Error())
return
}
userDataSlice = append(userDataSlice, userData)
}
fmt.Println(userDataSlice)
err = rows.Close()
if err != nil {
fmt.Println(err.Error())
}
}
运行结果:
[{1 johny 2019-07-08 14:05:40} {2 anson 2019-07-08 16:33:19}]
2)QueryRowx()
func (db *DB) QueryRowx(query string, args ...interface{}) *Row
代码示例
func queryRowx(Db *sqlx.DB) {
//定义结构体保存数据
type userinfo struct {
Uid int `db:"uid"`
UserName string `db:"username"`
CreateTime string `db:"create_time"`
}
var userData *userinfo = new(userinfo)
row := Db.QueryRowx("select uid, username, create_time from userinfo where uid = 1")
err := row.StructScan(userData)
if err != nil {
fmt.Printf("scan data failed, error is [%v]", err.Error())
}
fmt.Println(userData.Uid, userData.UserName, userData.CreateTime)
}
运行结果:
1 johny 2019-07-08 14:05:40
说了这么多,Query(),QueryRow() 不如 Get(),Select() 方法简洁
空值处理
Scan() 方法处理数据库中的 null
1)使用标准库中的数据类型
数据库中有一个特殊的类型,null 空值,可是 null 不能通过 scan 直接给变量赋值,也不能将 null 赋值给 nil,对于 null 必须指定特殊的类型,这些类型定义在 sqlx 扩展库中,例如 sql.NullFloat64,sql.NullString,sql.NullBool,sql.NullInt64,如果在扩展库中找不到匹配的值,可以尝试在驱动中寻找,下面的 demo,当数据表中 create_time 字段为 null 时,如果直接这样查询,会提示错误:
sql: Scan error on column index 2, name "create_time": unsupported Scan, storing driver.Value type <nil> into type *string
所以需要将代码改为:
func queryRow(Db *sqlx.DB) {
row := Db.QueryRow("select uid, username, create_time from userinfo")
var uid int
var userName string
var createTime sql.NullString
err := row.Scan(&uid, &userName, &createTime)
if err != nil {
fmt.Printf("scan data failed, error is [%v]", err.Error())
return
}
fmt.Println(uid, userName, createTime)
}
运行结果:
1 johny { false}
上面的运行结果中 { false},其实是 空字符串 与 string 类型的判断结果
在查询数据之前,查询结果有两种情况,null 与 非null,所以是需要验证的,如果值为 null,则会输出 NullString 的默认值,否则输出查询的值,demo 如下:
func queryRow(Db *sqlx.DB) {
row := Db.QueryRow("select uid, username, create_time from userinfo")
var uid int
var userName string
var createTime sql.NullString
err := row.Scan(&uid, &userName, &createTime)
if err != nil {
fmt.Printf("scan data failed, error is [%v]", err.Error())
return
}
fmt.Printf("%d %s\n", uid, userName)
fmt.Printf("createTime.String: '%v'\n", createTime.String)
fmt.Printf("createTime.Valid: %v\n", createTime.Valid)
}
运行结果:
//null值的情况
1 johny
createTime.String: ''
createTime.Valid: false
//值存在的情况
1 johny
createTime.String: '2019-07-08 12:53:18'
createTime.Valid: true
2)使用 []byte 接收数据
如果我们不关心查询的字段数据是不是 null 的时候,只是想把它当做空字符串处理就行,可以定义 []byte 接收数据,这样处理后,如果有值就获取值([]byte),如果没有则获取的为空字符串,demo 如下:
func queryRow(Db *sqlx.DB) {
row := Db.QueryRow("select uid, username, create_time from userinfo")
var uid []byte
var userName []byte
var createTime []byte
err := row.Scan(&uid, &userName, &createTime)
if err != nil {
fmt.Printf("scan data failed, error is [%v]", err.Error())
return
}
fmt.Printf("%s %s\n", uid, userName)
fmt.Printf("createTime.String: '%s'\n", createTime)
}
运行结果:
//有值的情况
1 johny
createTime.String: '2019-07-08 12:53:18'
//null值的情况
1 johny
createTime.String: ''
自动匹配字段数据
竟然所有的数据都能通过 []byte 进行接收,而字段名都是 string 类型,那么可以就可以把查询的数据放到 map 中保存,然后根据 key 进行取值,这样就方便多了
demo:
package main
import (
"fmt"
_ "github.com/go-sql-driver/mysql"
"github.com/jmoiron/sqlx"
) var (
userName string = "root"
password string = "seemmo"
ipAddrees string = "10.10.4.80"
port int = 3306
dbName string = "golang_db"
charset string = "utf8"
) func connectMysql() *sqlx.DB {
dsn := fmt.Sprintf("%s:%s@tcp(%s:%d)/%s?charset=%s", userName, password, ipAddrees, port, dbName, charset)
Db, err := sqlx.Open("mysql", dsn)
if err != nil {
fmt.Printf("mysql connect failed, detail is [%v]", err.Error())
}
return Db
} func queryData(Db *sqlx.DB) {
rows, err := Db.Query("select uid, username, create_time from userinfo")
if err != nil {
fmt.Printf("query data failed, error is [%v]", err.Error())
return
} cols, err := rows.Columns()
if err != nil {
fmt.Errorf("get rows columns failed, error is [%v]", err.Error())
} var vals = make([][]byte, len(cols)) //用来存放查询数据
var scanSlice = make([]interface{}, len(cols)) //用来当做参数,Scan 接收接口类型的参数
////将 []byte 放入接口
for i := range cols {
scanSlice[i] = &vals[i]
} var sliceMapData = make([]map[string]string, 0) for rows.Next() {
err := rows.Scan(scanSlice...)
if err != nil {
fmt.Printf("scan data failed, error is [%v]", err.Error())
return
} var mapData = make(map[string]string)
//这里遍历的是 字节切片
for i, value := range vals {
mapData[cols[i]] = string(value)
}
fmt.Println(mapData)
sliceMapData = append(sliceMapData, mapData)
}
fmt.Println(sliceMapData) err = rows.Close()
if err != nil {
fmt.Println(err.Error())
}
} func main (){
var Db *sqlx.DB = connectMysql()
defer Db.Close() queryData(Db)
} 运行结果:
map[create_time:2019-07-08 14:05:40 uid:1 username:johny]
map[create_time: uid:2 username:anson]
[map[create_time:2019-07-08 14:05:40 uid:1 username:johny] map[create_time: uid:2 username:anson]]
查询的是全部字段的数据,使用 rows.Columns() 方法可以获取到字段数据的切片([]string)
然后创建一个切片 vals,用来存放所取出来的数据结果
接下来又定义一个切片 scanSlice,在 Scan() 中使用,因为Scan() 方法接收的数据是接口类型,将数据库的查询结果复制给到它
vals 则得到了 scanSlice 复制给它的值,因为是 byte 切片,因此在循环一次,将其转换成 string,最后添加到 map 类型中
参考链接:https://www.cnblogs.com/zhaof/p/8509164.html
ending ~
Go 操作 Mysql(二)的更多相关文章
- Go基础之--操作Mysql(二)
在上一篇文章中主要整理了Golang连接mysql以及一些基本的操作,并进行了大概介绍,这篇文章对增删查改进行详细的整理 读取数据 在上一篇文章中整理查询数据的时候,使用了Query的方法查询,其实d ...
- python操作mysql二
游标 游标是一种能从包括多条数据记录的结果集中每次提取一条记录的机制,游标充当指针的作用,尽管游标能遍历结果中的所有行,但它一次只指向一行,游标的作用就是用于对查询数据库所返回的记录进行遍历,以便进行 ...
- MySQL之命令行简单操作MySQL(二)
一:命令行连接数据库 打开终端,运行命令mysql -uroot -p (p后面加密码,可以直接加,也可以回车在下一行输入,为了不暴露密码,回车在下行输入 退出:exit或quit 查看版本信息: s ...
- Python全栈开发之MySQL(二)------navicate和python操作MySQL
一:Navicate的安装 1.什么是navicate? Navicat是一套快速.可靠并价格相宜的数据库管理工具,专为简化数据库的管理及降低系统管理成本而设.它的设计符合数据库管理员.开发人员及中小 ...
- 一、初识MySQL数据库 二、搭建MySQL数据库(重点) 三、使用MySQL数据库 四、认识MySQL数据库的数据类型 五、操作MySQL数据库的数据(重点)
一.初识MySQL数据库 ###<1>数据库概述 1. 数据库 长期存储在计算机内的,由组织的可共享的数据集合 存储数据的仓库 文件 ...
- MySQL数据库(三)—— 表相关操作(二)之约束条件、关联关系、复制表
表相关操作(二)之约束条件.关联关系.复制表 一.约束条件 1.何为约束 除了数据类型以外额外添加的约束 2.约束条件的作用 为了保证数据的合法性,完整性 3.主要的约束条件 NOT NULL # ...
- python【第十二篇下】操作MySQL数据库以及ORM之 sqlalchemy
内容一览: 1.Python操作MySQL数据库 2.ORM sqlalchemy学习 1.Python操作MySQL数据库 2. ORM sqlachemy 2.1 ORM简介 对象关系映射(英语: ...
- python + docker, 实现天气数据 从FTP获取以及持久化(二)-- python操作MySQL数据库
前言 在这一节中,我们主要介绍如何使用python操作MySQL数据库. 准备 MySQL数据库使用的是上一节中的docker容器 “test-mysql”. Python 操作 MySQL 我们使用 ...
- Python操作MySQL数据库(二)
pymsql是Python中操作MySQL的模块,其使用方法和MySQLdb几乎相同. 下载安装: pip install pymysql 1.执行SQL语句 #!/usr/bin/env pytho ...
- 【翻译】MongoDB指南/CRUD操作(二)
[原文地址]https://docs.mongodb.com/manual/ MongoDB CRUD操作(二) 主要内容: 更新文档,删除文档,批量写操作,SQL与MongoDB映射图,读隔离(读关 ...
随机推荐
- 【Tomcat】Tomcat 原理架构(一)
Tomcat是什么 开源的 Java Web 应用服务器,实现了 Java EE(Java Platform Enterprise Edition)的部 分技术规范,比如 Java Servlet.J ...
- 【DataBase】Hsqldb与项目集成
Hsqldb与项目集成 进行模式下集成 可以集成进行(In-Process)模式下的hsqldb,由于数据存储在文件中,即使程序退出,数据也不会被销毁 采用jdbc集成的方式 package com. ...
- 【插件】thinkphp5+百度编辑器自定义上传
1 官方下载sdk 2 在引入编辑器页面.写入js // 百度编辑器 UE.Editor.prototype._bkGetActionUrl = UE.Editor.prototype.getActi ...
- 忏悔言情小说带来的意淫以及对治方法 (转自学佛网:http://www.xuefo.net/nr/article55/554935.html)
小时候,因为父母经常吵架,我觉得很孤独,一个人经常孤零零的,就喜欢一个人看书,大人的书难免里面有情情爱爱的内容,结果就很喜欢里面的深情的爱情故事,总是幻想自己有一段爱情.其实就是意淫的开始,所以后来学 ...
- python之参数解析模块argparse
2.7之后python不再对optparse模块进行扩展,python标准库推荐使用argparse模块对命令行进行解析. 简单入门 先来看个例子: argparse_test.py: import ...
- echarts移动端demo
说明:建议移动端使用的时候自己定制需要的东西,详情看官网 ECharts 效果图: 代码: <!DOCTYPE html> <html style="height: 1 ...
- angular2 ng-if
ng-if <td ><div class="td-li" > <a (click)="open(i)" class=" ...
- 09点睛Spring4.1-AOP
9.1 AOP AOP可以了让一组类共享相同的行为.在OOP中只能通过继承类和实现接口,这样使代码的耦合度增强,且类继承只能为单继承,阻碍更多行为添加到一组类上; 下面演示一个日志系统的实现,简单但不 ...
- DevOps - 持续集成(Continuous Integration)
1 - 持续集成简介 持续集成(Continuous integration,简称CI)是软件的开发和发布标准流程中最重要的部分. 作为一种开发实践,在CI中可以通过自动化等手段高频率地去获取产品反馈 ...
- Apache POI操作pptx基本使用
最近有一个ppt操作的需求,因此查了下相关的资料 ppt分类 (1)2007版之前的 是基于二进制的文件格式 细节没有完全公开,第三方厂商多是用单向工程方法猜测和分析出来的.WPS做得好一些,但开源的 ...