04使用 Go 连接数据库

简介

在 Go 中连接数据库的方式有很多, 这里我们选择使用 ORM 的方式,
也就不用写原生的 SQL 语句了.

Go 的 ORM 库也有很多, 这里选择了 gorm.

安装 gorm

go get -u github.com/jinzhu/gorm

启动数据库

数据库选择了最主流的 mysql.

创建数据库的方式有很多, 为了便于清理, 选择使用 docker 创建数据库.

新建一个 docker-compose.yml 文件, 在根目录下:

version: "3.7"

services:
  mysql:
    image: mysql:8
    command: --default-authentication-plugin=mysql_native_password
    environment:
      MYSQL_ROOT_PASSWORD: 1234
    ports:
      - 3306:3306

  adminer:
    image: adminer:4
    ports:
      - 8080:8080

  dbclient:
    image: mysql:8
    command: mysql -hmysql -uroot -p
    # source /home/script/db.sql
    # select * from tb_users \G;
    volumes:
      - ./script:/home/script

运行下面的命令在后台启动数据库:

docker-compose up -d mysql

使用下面的命令连接到 msyql 的 cli 上:

docker-compose run --rm dbclient

这会提示你输入密码, 密码是 1234 (在 mysql 的环境变量 MYSQL_ROOT_PASSWORD 中设置).

然后, 在 cli 中输入以下 SQL, 创建一个表格:

CREATE DATABASE IF NOT EXISTS `db_apiserver`;

接下来, 就可以完善 Go 代码了.

初始化数据库连接

在根目录下创建一个 model 文件夹, 这里定义数据模型, 以及数据库的初始连接.

在 model 目录下创建一个 init.go 文件, 用于初始化数据库连接.

package model

import (
    "fmt"

    "github.com/jinzhu/gorm"
    _ "github.com/jinzhu/gorm/dialects/mysql"
    "github.com/sirupsen/logrus"
    "github.com/spf13/viper"
)

type Database struct {
    Self *gorm.DB
}

// 单例
var DB *Database

func (db *Database) Init() {
    DB = &Database{
        Self: GetDB(),
    }
}

func (db *Database) Close() {
    DB.Self.Close()
}

func openDB(username, password, addr, name string) *gorm.DB {
    config := fmt.Sprintf(
        "%s:%s@tcp(%s)/%s?charset=utf8mb4&parseTime=%t&loc=%s",
        username,
        password,
        addr,
        name,
        true,
        // "Asia%2FShanghai",  // 必须是 url.QueryEscape 的
        "Local",
    )
    db, err := gorm.Open("mysql", config)
    if err != nil {
        logrus.Fatalf("数据库连接失败. 数据库名字: %s. 错误信息: %s", name, err)
    } else {
        logrus.Infof("数据库连接成功, 数据库名字: %s", name)
    }

    setupDB(db)
    return db
}

func setupDB(db *gorm.DB) {
    db.LogMode(viper.GetBool("gormlog"))
    // 用于设置最大打开的连接数,默认值为0表示不限制.设置最大的连接数,可以避免并发太高导致连接mysql出现too many connections的错误。
    //db.DB().SetMaxOpenConns(20000)
    // 用于设置闲置的连接数.设置闲置的连接数则当开启的一个连接使用完成后可以放在池里等候下一次使用。
    db.DB().SetMaxIdleConns(0)
}

func InitDB() *gorm.DB {
    return openDB(
        viper.GetString("db.username"),
        viper.GetString("db.password"),
        viper.GetString("db.addr"),
        viper.GetString("db.name"),
    )
}

func GetDB() *gorm.DB {
    return InitDB()
}

注意导入的时候, 需要导入对应的数据库驱动, mysql 需要如下的导入:

"github.com/jinzhu/gorm"
_ "github.com/jinzhu/gorm/dialects/mysql"

主要是定义了一个结构来保存数据库实例 *gorm.DB, 用到了单例模式:

type Database struct {
    Self *gorm.DB
}

// 单例
var DB *Database

然后定义了数据库的初始化方法和关闭方法.

打开的时候用到了配置文件中的参数, 需要在配置文件 config.yaml 中添加如下的参数:

db:
  name: db_apiserver
  addr: 127.0.0.1:3306
  username: root
  password: "1234"

服务器启动时连接数据库

在 runServer 函数中添加下面的代码:

// 初始化数据库
model.DB.Init()
defer model.DB.Close()

总结

其实连接数据库的问题并不太大, 以前觉得麻烦是因为在本地启动一个数据库麻烦,
但是在有了 docker 之后, 一切就变得简单可重复了, 再也不用担心兼容性了.

当前部分的代码

作为版本 v0.4.0

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容

  • feisky云计算、虚拟化与Linux技术笔记posts - 1014, comments - 298, trac...
    不排版阅读 3,843评论 0 5
  • ORA-00001: 违反唯一约束条件 (.) 错误说明:当在唯一索引所对应的列上键入重复值时,会触发此异常。 O...
    我想起个好名字阅读 5,300评论 0 9
  • MySQL物理文件组成——日志文件——错误日志(Error Log) 错误日志记录了MyQL服务器运行过程中所有较...
    每天学点编程阅读 2,252评论 0 12
  • 在山脚下 憧憬着山顶的景 于是—一步一个脚印 一点一滴的汗水流下 为了山顶的景 流再多汗也无妨 在山腰上 憧憬着山...
    雨落大西洋31号阅读 200评论 0 0
  • 首先是观看了河南濮阳油田一小孩子排练的童话剧,然后就鼓励孩子参加。老师给同学们写信,然后告诉孩子们,能够挑战自己,...
    王辉霞阅读 177评论 0 0