百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

Go语言零到一:数据库交互_go语言数据库连接池

itomcoil 2025-09-04 07:59 4 浏览

引言

Go 语言的 database/sql 包为开发者提供了一个抽象层,使得与 MySQL 数据库的交互变得更加容易。

1. 安装 MySQL 驱动

为了能够与 MySQL 数据库通信,我们需要安装 go-sql-driver/mysql 驱动。可以通过 Go 的包管理工具 go get 安装:

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

2. 创建数据库连接

使用 sql.Open 方法来打开一个数据库连接,并使用 Ping 方法测试连接。

package main

import (
	"database/sql"
	"log"

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

func main() {
	db, err := sql.Open("mysql", "user:password@tcp(127.0.0.1:3306)/dbname")
	if err != nil {
		log.Fatal(err)
	}
	defer db.Close()

	err = db.Ping()
	if err != nil {
		log.Fatal(err)
	}
	log.Println("Connected to the database successfully.")
}

3. 执行SQL语句

3.1 查询单条记录

使用 db.QueryRow 方法来查询一条记录,并通过 Scan 方法将结果赋值给变量。

var id int
var name string
err := db.QueryRow("SELECT id, name FROM users WHERE id = ?", 1).Scan(&id, &name)
if err != nil {
  if errors.Is(err, sql.ErrNoRows) {
    log.Println("No rows were returned!")
  } else {
    log.Fatal(err)
  }
}
log.Printf("Found user with ID %d and name %s\n", id, name)

3.2 查询多条记录

使用 db.Query 方法来查询多条记录,并遍历结果集。

rows, err := db.Query("SELECT id, name FROM users")
if err != nil {
  log.Fatal(err)
}
defer rows.Close()

for rows.Next() {
  var id int
  var name string
  err := rows.Scan(&id, &name)
  if err != nil {
    log.Fatal(err)
  }
  log.Printf("User ID: %d, Name: %s\n", id, name)
}

3.3 插入记录

使用 db.Exec 方法来插入记录,并获取最后插入的 ID 和受影响的行数。

result, err := db.Exec("INSERT INTO users (name, email) VALUES (?, ?)", "John Doe", "john.doe@example.com")
if err != nil {
  log.Fatal(err)
}

lastInsertId, err := result.LastInsertId()
if err != nil {
  log.Fatal(err)
}

rowsAffected, err := result.RowsAffected()
if err != nil {
  log.Fatal(err)
}

log.Printf("Last Insert ID: %d, Rows Affected: %d\n", lastInsertId, rowsAffected)

3.4 更新记录

同样使用 db.Exec 方法来更新记录。

_, err := db.Exec("UPDATE users SET name = ? WHERE id = ?", "Jane Doe", 1)
if err != nil {
  log.Fatal(err)
}

3.5 删除记录

还是使用 db.Exec 方法来删除记录。

_, err := db.Exec("DELETE FROM users WHERE id = ?", 1)
if err != nil {
  log.Fatal(err)
}

3.6 事务处理

在进行涉及多个操作的业务逻辑时,事务处理是必不可少的。事务保证了操作的原子性、一致性、隔离性和持久性(ACID)。

tx, err := db.Begin()
if err != nil {
  log.Fatal(err)
}

// 执行多个操作
_, err = tx.Exec("UPDATE users SET balance = balance - 10 WHERE id = ?", 1)
if err != nil {
  tx.Rollback()
  log.Fatal(err)
}

_, err = tx.Exec("INSERT INTO logs (user_id, action) VALUES (?, 'debit')", 1)
if err != nil {
  tx.Rollback()
  log.Fatal(err)
}

err = tx.Commit()
if err != nil {
  log.Fatal(err)
}

3.7 错误处理

在与数据库交互时,错误处理非常重要。始终检查 ExecQuery 返回的错误,并在适当的情况下回滚事务。

if err != nil {
  if rbErr := tx.Rollback(); rbErr != nil {
    log.Fatalf("Failed to rollback transaction: %v", rbErr)
  }
  log.Fatalf("Error executing statement: %v", err)
}

4. 示例代码

下面是一个完整的示例,展示了如何使用 Go 语言与 MySQL 数据库进行交互,包括连接、查询、插入、更新和删除记录。

package main

import (
	"database/sql"
	"log"

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

func main() {
	db, err := sql.Open("mysql", "user:password@tcp(127.0.0.1:3306)/dbname")
	if err != nil {
		log.Fatal(err)
	}
	defer db.Close()

	err = db.Ping()
	if err != nil {
		log.Fatal(err)
	}
	log.Println("Connected to the database successfully.")

	// 查询记录
	rows, err := db.Query("SELECT id, name FROM users")
	if err != nil {
		log.Fatal(err)
	}
	defer rows.Close()

	for rows.Next() {
		var id int
		var name string
		err := rows.Scan(&id, &name)
		if err != nil {
			log.Fatal(err)
		}
		log.Printf("User ID: %d, Name: %s\n", id, name)
	}

	// 插入记录
	result, err := db.Exec("INSERT INTO users (name, email) VALUES (?, ?)", "John Doe", "john.doe@example.com")
	if err != nil {
		log.Fatal(err)
	}

	lastInsertId, err := result.LastInsertId()
	if err != nil {
		log.Fatal(err)
	}

	rowsAffected, err := result.RowsAffected()
	if err != nil {
		log.Fatal(err)
	}

	log.Printf("Last Insert ID: %d, Rows Affected: %d\n", lastInsertId, rowsAffected)

	// 更新记录
	_, err = db.Exec("UPDATE users SET name = ? WHERE id = ?", "Jane Doe", 1)
	if err != nil {
		log.Fatal(err)
	}

	// 删除记录
	_, err = db.Exec("DELETE FROM users WHERE id = ?", 1)
	if err != nil {
		log.Fatal(err)
	}
}

5. 数据库推荐

  • SQL 数据库
  1. MySQL
    1. go-sql-driver/mysql:官方 MySQL 驱动,支持基本的 CRUD 操作。
    2. gorm/gorm:一个 ORM 库,支持 MySQL、PostgreSQL、SQLite 等多种数据库。
    3. jinzhu/gorm:另一个流行的 ORM 库,同样支持多种数据库。
  2. PostgreSQL
    1. lib/pq:官方 PostgreSQL 驱动。
    2. jmoiron/sqlx:扩展了 database/sql API,提供了结构化的查询结果处理。
  3. SQLite
    1. mattn/go-sqlite3:Go 语言的 SQLite 驱动。
  4. SQL Server
    1. microsoft/go-mssqldb:Microsoft 提供的 SQL Server 驱动。
  • NoSQL 数据库
    • MongoDB
    1. mongo-go-driver/mongo:官方 MongoDB 驱动。
    2. go.mongodb.org/mongo-driver:官方推荐的新版驱动。
  1. Redis
    1. go-redis/redis:一个高性能的 Redis 客户端库。
    2. goredis/goredis:早期版本的 Redis 客户端库。
  2. Elasticsearch
      1. elastic/go-elasticsearch:Elasticsearch 官方 Go 客户端。

#数据库##golang##MySQL##SQL#

相关推荐

NAS下搭建FastGpt,一个基于 LLM 大语言模型的知识库问答系统

本内容来源于@什么值得买APP,观点仅代表作者本人|作者:熊猫不是猫QAQ前言FastGPT是一个基于LLM大语言模型的知识库问答系统,提供开箱即用的数据处理、模型调用等能力。同时可以通过Flow可...

MongoDB入门实操《二》_mongodb从入门到商业实战

常规命令使用首先我们来了解几个概念,虽然MongoDB入门实操《上篇》这篇文章已经提到过,这里再次加深印象:集合:Mongo中的集合就是mysql的表的表现形式文档:文档的数据结构和JSON基本...

Go语言零到一:数据库交互_go语言数据库连接池

引言Go语言的database/sql包为开发者提供了一个抽象层,使得与MySQL数据库的交互变得更加容易。1.安装MySQL驱动为了能够与MySQL数据库通信,我们需要安装go...

Java中使用MongoDB数据库_java操作mongo

一、Java实现对MongDB的操作1、前提条件除了通过启动mongo进程进如Shell环境访问数据库外,MongoDB还提供了其他基于编程语言的访问数据库方法。MongoDB官方提...

Spring Boot 集成 MongoDB:从入门到生产实践

1、MongoDB简介MongoDB是一个开源的、面向文档的NoSQL数据库,旨在提供高性能、高可用性和易扩展性。它与传统的关系型数据库(如MySQL)有着根本性的区别。面向文档(Docum...

发现一个不错的库推荐给大家:DuckDB + Python + Pandas 量化研究利器

大家好,我是花姐。最近折腾数据处理的时候,发现了一个宝藏库——DuckDB,配合Python和Pandas简直不要太香!今天我就聊聊它到底有多好用,特别适合我们搞量化研究的同学。1.Duck...

SpringBoot 自研「轻量级 API 防火墙」:单机内嵌,支持在线配置

1.背景与痛点在做后端开发时,我们常常会遇到这样的困境:接口被恶意刷流量:比如某个查询接口被短时间大量调用,数据库连接数打满,最终拖垮整个服务。缺少细粒度防护能力:很多系统只有粗糙的全局限流,但某些...

福建新画卷,把福建成绩“画”给你看

·x-_p-_o-[U2FsdGVkX1/DHR1fwp5qMkbHvRjusk9BgvDHM/8zbH+7Z+bZnW0jD3Vk67FnKGbJHp2L4vKFqg7ryQ7Zadr0+p+82E...

[续]江西话汉语赣方言之叠词_江西话什么意思

一、赣语耒阳话每次去湖南耒阳,老俵听到我是江西人,都分外亲切:“我们祖上是从江西迁过来的!”在得知明朝初年湖广因战乱赤地千里,耒阳人祖上大都从江西永新迁来后,我就特别留意耒阳方言:耒阳话不仅保留了老...

SpringBoot 在线依赖包漏洞扫描仪,一键发现潜在安全漏洞

在日常项目开发中,依赖第三方库几乎是不可避免的。从spring-boot-starter到MyBatis、Log4j,再到各种工具类库,几乎每个应用都需要几十甚至上百个依赖。然而,这些依赖并非完...

一文学会Python的变量命名规则!_简述python变量命名规则

目录1.变量的命名原则3.内置函数尽量不要做变量4.删除变量和垃圾回收机制5.结语1.变量的命名原则①由英文字母、_(下划线)、或中文开头②变量名称只能由英文字母、数字、下画线或中文字所组成。③英文字...

Python中8种Functools使用方法_python functools.partial

在本文中,我们来看看functools标准库模块以及您可以用它做的6件很酷的事情1.缓存可以使用@cache装饰器(以前称为@lru_cache)作为“简单的轻量级无界函数缓存”。典型的例子是...

每天5分钟,python 速成(59)_python速成视频教程

首先明确学习目标,今天的目标是python中模块学习模块模块就好比是工具包,要想使用这个工具包中的工具(就好比函数),就需要导入这个模块importrandomimportmath#数学相...

珍藏版-11款爱心的免费公布的数学方程式和可视化-python版

一,图:1直角坐标系(9款):2追加一款:直角坐标系:3极坐标系(1款):二,爱心的数学方程式:1直角坐标系:1.1方程式1:Love-1进行转换:python代码中需要进行修改为:x=...

12种不同方法教你用Python性能优化!(建议收藏)

前言:今天为大家带来的内容是12种不同方法教你用Python性能优化!希望通过本文的内容能够帮助到各位!(代码块是用图片的方式呈现出来)1.减少冗余数据如用上三角或下三角的方式去保存一个大的对称矩阵。...