GO分层架构【4】Repository获取 *gorm.DB

Repository(或 Infra) 要如何获取 *gorm.DB 对象的?是通过全局变量,还是通过结构体注入?

在 Go 的 GORM 开发中,获取 *gorm.DB 对象主要有两种流派:"简单粗暴型"和"工程严谨型"。

作为一名 Go 开发者,理解这两者的区别对你调整架构至关重要:

1. 全局变量模式(简单粗暴型)

这是很多教程里的写法。在 infra 包里定义一个全局变量 DB

  • 代码长这样:

    复制代码
    // internal/infra/db.go
    var DB *gorm.DB
    
    func InitDB() {
        db, _ := gorm.Open(...)
        DB = db
    }
  • 如何调用: 在任何地方(Handler/Logic/Repo)直接调用 infra.DB.Find(&user)

  • 评价:

    • 优点:写起来极快,不用传参。
    • 缺点:极其难测。如果你想写单元测试,或者想连接两个不同的数据库,全局变量会让你痛不欲生。而且这种隐藏的依赖关系会导致代码越写越乱。

2. 结构体注入模式(工程严谨型 - 推荐)

这是主流微服务(如 go-zero, kratos)和互联网大厂的写法。

  • 代码逻辑:

    1. infra 负责生成 *gorm.DB 并返回。
    2. repository 定义一个结构体,把 *gorm.DB 存进去。
    3. logic 调用 repository 的实例。
  • 代码示例:

    复制代码
    // internal/repository/user_repo.go
    type UserRepo struct {
        db *gorm.DB // 私有变量,不暴露给外面
    }
    
    // 构造函数,创建 Repo 时必须传入 DB
    func NewUserRepo(db *gorm.DB) *UserRepo {
        return &UserRepo{db: db}
    }
    
    func (r *UserRepo) FindByID(id uint) (*model.User, error) {
        var user model.User
        err := r.db.First(&user, id).Error
        return &user, err
    }

3. 为什么推荐"结构体注入"?

  1. 链路清晰:你在 main.go 启动时就能一眼看出:DB 传给了 Repo,Repo 传给了 Logic。
  2. 方便做"假"测试(Mock):写单元测试时,你可以传一个模拟的 DB 进去,而不需要真的连数据库。
  3. 支持多库:如果你以后业务大了,读写分离或者有多个库,你只需要在初始化时传入不同的 DB 实例即可,逻辑层代码一行都不用改。

4. 结合你之前的架构:如何串联起来?

如果你打算采用推荐的写法,你的 main.go 启动流程大致如下:

  1. 加载配置:cfg := config.Load()
  2. 初始化基础设施:db := infra.NewDB(cfg) (返回 *gorm.DB)
  3. 创建 Repository:userRepo := repository.NewUserRepo(db)
  4. 创建 Logic:userLogic := logic.NewUserLogic(userRepo)
  5. 注册 Handler:r.POST("/user", handler.NewUserHandler(userLogic))

相关推荐
我叫黑大帅5 小时前
其实跨域问题是后端来解决的? CORS
后端·面试·go
审判长烧鸡9 小时前
GO分层架构【2】使用GIN与GORM
go·分层架构
Go_error1 天前
Go channel 数据聚合
后端·go
stark张宇1 天前
Go 语言实现安全的分享链接:AES 加密 + SHA256 签名 + 过期防重放
后端·go
我叫黑大帅1 天前
Golang中的map的key可以是哪些类型?可以嵌套map吗?
后端·面试·go
用户095367515832 天前
Go :如何声明变量(var)与常量(const)
后端·go
FelixBitSoul2 天前
Go 语言面试深度全攻略:从工程化到底层原理,一文通杀
后端·go
你有医保你先上2 天前
Elasticsearch Go 客户端
后端·elasticsearch·go
Vale3653 天前
[Go]字符串 比较
go