当前位置:首页 > 技术 > 正文内容

Go语言GORM多对多关系实战详解

访客 技术 2026年5月24日 3

1. 数据库表设计与数据准备

首先创建三张表:用户表、资料表和关联中间表。

用户表(user):

CREATE TABLE `user` (
  `id` bigint(20) NOT NULL,
  `user_key` bigint(20) NOT NULL,
  `account` char(32) NOT NULL
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

初始化数据(为方便测试,user_key和id保持一致):

+----+----------+---------+
| id | user_key | account |
+----+----------+---------+
|  1 |        1 | user-1  |
+----+----------+---------+

资料表(profile):

CREATE TABLE `profile` (
  `id` bigint(20) NOT NULL,
  `profile_key` bigint(20) NOT NULL,
  `desc` char(32) NOT NULL
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

初始化数据(同理,profile_key与id保持一致):

+----+-------------+-----------+
| id | profile_key | desc      |
+----+-------------+-----------+
|  2 |           2 | profile-1 |
|  3 |           3 | profile-2 |
+----+-------------+-----------+

关联表(user_profile):用于存储多对多关系,一个用户能关联多个资料,一个资料也能被多个用户关联。

CREATE TABLE `user_profile` (
  `id` bigint(20) NOT NULL,
  `profile_id` bigint(20) NOT NULL,
  `user_id` bigint(20) NOT NULL
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

关系数据:

+----+------------+---------+
| id | profile_id | user_id |
+----+------------+---------+
|  4 |          2 |       1 |
|  5 |          3 |       1 |
+----+------------+---------+

2. 标签字段含义速查表

标签项说明
many2many指定关联中间表的表名,若不指定则GORM默认使用两张表名+下划线组合
foreignKey主表中用于关联的字段,对应中间表的joinForeignKey
joinForeignKey中间表中与主表foreignKey字段对应的列
references目标表中用于关联的字段,对应中间表的joinReferences
joinReferences中间表中与目标表references字段对应的列

3. Go代码实现

package test_service

import (
	"encoding/json"
	"fmt"
	"gorm.io/driver/mysql"
	"gorm.io/gorm"
	"gorm.io/gorm/logger"
)

// User 用户模型
type User struct {
	Id      int    `json:"id" gorm:"id"`
	UserKey int    `json:"user_key" gorm:"user_key"`
	Account string `json:"account" gorm:"account"`

	// 方式一:使用默认主键关联
	// 系统会用 user.id → user_profile.user_id,profile.id → user_profile.profile_id
	Profiles []Profile `gorm:"many2many:user_profile;joinForeignKey:user_id;joinReferences:profile_id;" json:"profiles"`

	// 方式二:使用自定义字段关联
	// 通过 foreignKey:user_key 指定 user.user_key 对应 user_profile.user_id
	// 通过 references:profile_key 指定 profile.profile_key 对应 user_profile.profile_id
	AllProfiles []Profile `gorm:"many2many:user_profile;foreignKey:user_key;joinForeignKey:user_id;references:profile_key;joinReferences:profile_id;" json:"all_profiles"`
}

func (*User) TableName() string {
	return "user"
}

// Profile 资料模型
type Profile struct {
	Id         int    `json:"id" gorm:"id"`
	ProfileKey int    `json:"profile_key" gorm:"profile_key"`
	Desc       string `json:"desc" gorm:"desc"`
}

func (*Profile) TableName() string {
	return "profile"
}

// UserProfile 关联表模型
type UserProfile struct {
	Id        int `json:"id" gorm:"id"`
	ProfileId int `json:"profile_id" gorm:"profile_id"`
	UserId    int `json:"user_id" gorm:"user_id"`
}

func (*UserProfile) TableName() string {
	return "user_profile"
}

var db *gorm.DB

func init() {
	addr := "127.0.0.1:3306"
	user := "user"
	pass := "pass"
	name := "test_1"
	dsn := "%s:%s@tcp(%s)/%s?charset=utf8mb4&parseTime=True&loc=Local"
	dsn = fmt.Sprintf(dsn, user, pass, addr, name)
	db, _ = gorm.Open(mysql.Open(dsn), &gorm.Config{
		Logger: logger.Default.LogMode(logger.Info),
	})
}

// ManyToMany 执行多对多查询
func ManyToMany() {
	var users []User
	// Preload 实现关联查询,可同时加载多个关联关系
	db.Preload("Profiles").Preload("AllProfiles").Where("id", 1).Find(&users)
	by, _ := json.Marshal(users)
	fmt.Println(string(by))
}

4. 查询结果示例

[
    {
        "id": 1,
        "user_key": 1,
        "account": "user-1",
        "profiles": [
            {
                "id": 2,
                "profile_key": 2,
                "desc": "profile-1"
            },
            {
                "id": 3,
                "profile_key": 3,
                "desc": "profile-2"
            }
        ],
        "all_profiles": [
            {
                "id": 2,
                "profile_key": 2,
                "desc": "profile-1"
            },
            {
                "id": 3,
                "profile_key": 3,
                "desc": "profile-2"
            }
        ]
    }
]

两种关联方式均正确返回了用户关联的两条资料记录。

5. 底层SQL执行流程

GORM在执行Preload时会分三步完成查询:

-- 第一步:查询主表记录
SELECT * FROM `user` WHERE `id` = 1;

-- 第二步:根据主表id查询中间表
SELECT * FROM `user_profile` WHERE `user_profile`.`user_id` = 1;

-- 第三步:根据中间表的profile_id查询目标表
SELECT * FROM `profile` WHERE `profile`.`id` IN (2,3);

通过这种分步查询策略,GORM能高效地完成多对多关系的预加载,避免N+1查询问题。

相关文章

Linux crontab 详解

1) crontab 是什么cron 是 Linux 的定时任务守护进程;crontab 是用来编辑/查看“按时间周期执行命令”的表(cron table)。常见两类:用户 crontab:每个用户一份(crontab -e 编辑)系统级 crontab / cron.d:可指定执行用户(/etc/crontab、/etc/cron.d/*)2) crontab 时间...

富文本里可以允许的 HTML 属性

一、所有标签默认允许的安全属性(极少)class        (可选)id           (通常建议禁用)title️ 注意:id 容易被滥用做锚点注入,很多系统直接禁用class 允许的话最好只允许固定前缀(如 editor-*)二、a 标签允许属性<a href="" t...

Mac 安装 Node.js 指南

方法一:通过官网安装包(最简单,适合初学者)如果你只是想快速安装并开始使用,这是最直接的方法。访问 Node.js 官网。页面会显示两个版本:LTS (Recommended For Most Users):长期支持版,最稳定。建议选这个。Current:最新特性版,包含最新功能但可能不够稳定。下载 .pkg 安装包并运行。按照安装向导点击“下一步”即可完成。方法二:使用 Homebrew 安装(...

Dom\HTML_NO_DEFAULT_NS 的副作用:自动加闭合标签

在使用Dom\HTMLDocument时,Dom\HTML_NO_DEFAULT_NS 将禁止在解析过程中设置元素的命名空间, 此设置是为了与DOMDocument向后兼容而存在的。当使用它时,已知的一个副作用就是:自动加闭合标签例如 </img> 为什么会这样?当你使用:Dom\HTML_NO_DEFAULT_NS文档会变成 无命名空间模式,此时内部更接近 XML...

Laravel 事件和监听器创建

在 Laravel 中,使用 Artisan 命令创建 Events(事件) 和 Listeners(监听器) 是非常高效的。你可以通过以下几种方式来实现:1. 手动创建单个 Event如果你只想创建一个事件类,可以使用 make:event 命令:Bashphp artisan make:event UserRegistered执行后,文件将生成在 app/Even...

自定义域名解析神器 dnsmasq

什么是 dnsmasq?dnsmasq 是一个轻量级、功能强大的网络服务工具,专为小型和中等规模网络设计。它是一个综合的网络基础设施解决方案[1]。dnsmasq 能做什么?功能说明应用场景DNS 转发与缓存将 DNS 查询转发到上游服务器(ISP、Google DNS 等),并在本地缓存结果加快 DNS 查询速度,减少外部 DNS 流量本地 DNS解析本地网络设备的主机名,无需编辑&n...

发表评论

访客

◎欢迎参与讨论,请在这里发表您的看法和观点。