Golang GORM系列:GORM 高级查询教程
有效的数据检索是任何程序功能的基础。健壮的Go对象关系映射包(称为GORM)除了标准的CRUD操作之外,还提供了复杂的查询功能。这是学习如何使用GORM进行高级查询的综合资源。我们将涵盖WHERE条件、连接、关联、预加载相关数据,甚至涉及原始SQL查询。到最后,你的Go应用程序将具备以无与伦比的精度提取和操作数据的能力。
GORM where 条件
使用WHERE条件优化查询对于提取特定的数据子集至关重要。
步骤1:基本的WHERE子句
使用GORM的‘ Where ’方法来应用条件:
var expensiveProducts []Product
db.Where("price > ?", 50).Find(&expensiveProducts)
步骤2:AND & OR条件
使用逻辑运算符组合多个条件:
var filteredProducts []Product
db.Where("price > ? AND category = ?", 50, "Electronics").Find(&filteredProducts)
GORM中的连接和关联
模型之间的关联支持跨多个表的复杂查询。
步骤1:定义关联
在你的模型结构中建立关联:
type User struct {gorm.ModelOrders []Order
}type Order struct {gorm.ModelUserID uintProduct string
}
步骤2:执行连接
使用GORM的‘ Joins ’方法从关联的模型中检索数据:
var usersWithOrders []User
db.Joins("JOIN orders ON users.id = orders.user_id").Find(&usersWithOrders)
GORM预加载相关数据
有效地加载相关数据以最小化数据库查询。
步骤1:预加载关联
使用GORM的‘ Preload ’方法来快速加载相关数据:
var users []User
db.Preload("Orders").Find(&users)
步骤2:嵌套预加载
预加载嵌套关联,用于全面的数据检索;
var users []User
db.Preload("Orders.OrderItems").Find(&users)
GORM中的原始SQL查询
对于复杂的查询,GORM允许执行原始SQL语句。
步骤1:原始SQL查询
使用GORM的‘ raw ’方法执行原始SQL查询:
var products []Product
db.Raw("SELECT * FROM products WHERE price > ?", 50).Scan(&products)
步骤2:绑定变量
使用绑定变量进行更安全、更高效的查询:
var categoryName = "Electronics"
var expensivePrice = 100
var filteredProducts []Product
db.Raw("SELECT * FROM products WHERE category = ? AND price > ?", categoryName, expensivePrice).Scan(&filteredProducts)
完整示例
在现实场景中,用户有多个订单,一个订单可能包含多个产品。为了实现这一点,我们需要引入一个中间表(即关联表)来表示订单和产品之间的多对多关系。
数据模型如下
- User 模型:表示用户。
- Order 模型:表示订单。
- Product 模型:表示产品。
- OrderProduct 模型:表示订单和产品之间的多对多关系(中间表)。
完整代码
我们通过引入中间表 order_products
,实现了订单和产品之间的多对多关系。使用 Preload
方法可以高效地加载嵌套关系,避免多次查询数据库。这个实例展示了如何在 GORM 中处理复杂的多对多关系,并支持一个订单包含多个产品的场景。你可以根据实际需求进一步扩展模型和查询逻辑,例如添加更多的字段或条件。
package mainimport ("fmt""gorm.io/driver/mysql""gorm.io/gorm"
)// User 模型
type User struct {ID uintName stringEmail stringOrders []Order // 一个用户有多个订单
}// Order 模型
type Order struct {ID uintUserID uintUser User // 订单属于一个用户Products []Product `gorm:"many2many:order_products;"` // 一个订单有多个产品
}// Product 模型
type Product struct {ID uintName stringPrice float64Orders []Order `gorm:"many2many:order_products;"` // 一个产品可以属于多个订单
}// OrderProduct 模型(中间表)
type OrderProduct struct {OrderID uintProductID uintQuantity int // 订单中某个产品的数量
}func main() {// 连接数据库dsn := "user:password@tcp(127.0.0.1:3306)/dbname?charset=utf8mb4&parseTime=True&loc=Local"db, err := gorm.Open(mysql.Open(dsn), &gorm.Config{})if err != nil {panic("failed to connect database")}// 自动迁移模型db.AutoMigrate(&User{}, &Order{}, &Product{}, &OrderProduct{})// 插入测试数据user := User{Name: "John Doe", Email: "john@example.com"}db.Create(&user)product1 := Product{Name: "Laptop", Price: 1200.00}product2 := Product{Name: "Mouse", Price: 25.00}db.Create(&product1)db.Create(&product2)order := Order{UserID: user.ID}db.Create(&order)// 添加产品到订单db.Model(&order).Association("Products").Append([]Product{product1, product2})// 查询用户及其订单和产品信息var users []Userdb.Preload("Orders.Products").Find(&users)// 打印结果for _, user := range users {fmt.Printf("User: %s, Email: %s\n", user.Name, user.Email)for _, order := range user.Orders {fmt.Printf(" Order ID: %d\n", order.ID)for _, product := range order.Products {fmt.Printf(" Product: %s, Price: %.2f\n", product.Name, product.Price)}}}
}
数据模型
- Order 和 Product 的多对多关系:
- 一个订单可以包含多个产品,一个产品也可以属于多个订单。
- 使用
gorm:"many2many:order_products;"
标签定义多对多关系,order_products
是中间表的名称。
- 中间表 OrderProduct:
- 中间表包含
OrderID
和ProductID
作为外键,以及额外的字段Quantity
表示订单中某个产品的数量。
- 中间表包含
查询数据
- 使用
Preload
方法预加载嵌套关系:Preload("Orders")
加载用户的订单。Preload("Orders.Products")
加载每个订单的产品。
- 这样可以避免 N+1 查询问题,提高查询效率。
中间表数据
// 添加产品到订单
db.Model(&order).Association("Products").Append([]Product{product1, product2})
Association("Products")
:- 这里使用了 GORM 的
Association
方法,表示操作Order
模型与Product
模型之间的多对多关系。 Products
是Order
模型中定义的关联字段。
- 这里使用了 GORM 的
Append([]Product{product1, product2})
:Append
方法用于将产品添加到订单中。- 这里传入了两个产品:
product1
和product2
。
- 中间表数据的插入:
- 当调用
Append
方法时,GORM 会自动在中间表order_products
中插入数据。 - 插入的数据包括:
OrderID
:当前订单的 ID。ProductID
:每个产品的 ID。Quantity
:如果中间表有其他字段(如Quantity
),可以通过额外配置插入数据(见下文)。
- 当调用
假设:
- 订单的 ID 是
1
。 - 产品的 ID 分别是
1
(Laptop)和2
(Mouse)。
调用 Append
方法后,GORM 会自动在 order_products
表中插入以下数据:
OrderID | ProductID | Quantity |
---|---|---|
1 | 1 | 0 |
1 | 2 | 0 |
注意:如果中间表有其他字段(如
Quantity
),需要额外处理。
如果需要插入 Quantity
字段
如果中间表 OrderProduct
包含 Quantity
字段,并且希望在插入时设置数量,可以通过以下方式实现:
修改后的代码
// 添加产品到订单,并设置数量
orderProduct1 := OrderProduct{OrderID: order.ID, ProductID: product1.ID, Quantity: 1}
orderProduct2 := OrderProduct{OrderID: order.ID, ProductID: product2.ID, Quantity: 2}
db.Create(&orderProduct1)
db.Create(&orderProduct2)
- 手动创建
OrderProduct
记录,并设置OrderID
、ProductID
和Quantity
。 - 使用
db.Create
将记录插入到order_products
表中。
最后总结
从Go应用程序获取和修改数据的最全面的工具集是由GORM复杂的查询功能提供的。通过学习如何使用连接和关系、预加载相关数据,甚至尝试原始SQL查询,可以很快掌握精确而复杂地数据查询。这些特性不仅提高了程序的效率,而且还提供了对以前难以想象的复杂数据情况的访问。
相关文章:

Golang GORM系列:GORM 高级查询教程
有效的数据检索是任何程序功能的基础。健壮的Go对象关系映射包(称为GORM)除了标准的CRUD操作之外,还提供了复杂的查询功能。这是学习如何使用GORM进行高级查询的综合资源。我们将涵盖WHERE条件、连接、关联、预加载相关数据,甚至涉…...
智能GUI Agent是什么,有什么应用领域
智能GUI Agent是什么 研究背景与目的:GUI长期主导人机交互,LLM特别是多模态模型的出现,为GUI自动化带来变革,催生了基于LLM的GUI智能体。这些智能体可理解自然语言指令,处理复杂GUI元素并执行操作,改变了用户与软件交互方式。论文旨在梳理该领域发展脉络,剖析关键要素,…...
k8s优雅操作pod容器组
k8s优雅操作pod容器组 回退备份 kubectl get deploy deployName -o yaml>>deployName-bak-date "%Y-%m-%d".yaml获取副本数 replicasecho | kubectl get -o template deploy/deployName --template{{.spec.replicas}}停止容器组 kubectl scale deployment …...

在 Mac ARM 架构上使用 nvm 安装 Node.js 版本 16.20.2
文章目录 1. 安装 nvm(如果还没有安装的话)2. 加载 nvm 配置3. 列出特定系列的 Node.js 版本(远程):4. 安装 Node.js 16.20.25. 使用指定版本的 Node.js6. 验证安装 在 Mac ARM 架构上使用 nvm 安装 Node.js 版本 16.…...

MySQL创建存储过程和存储函数
【图书推荐】《MySQL 9从入门到性能优化(视频教学版)》-CSDN博客 《MySQL 9从入门到性能优化(视频教学版)(数据库技术丛书)》(王英英)【摘要 书评 试读】- 京东图书 (jd.com) MySQL9数据库技术_夏天又到了…...

PyQt学习记录03——批量设置水印
0. 目录 PyQt学习记录01——加法计算器 PyQt学习记录02——串口助手 1. 前言 本次主要是为了学习Qt中的 QFileDialog 函数, QFileDialog.getExistingDirectory:用于选择文件夹,返回的是一个文件夹路径。 QFileDialog.getOpenFileName&…...

vivo手机和Windows电脑连接同一个WiFi即可投屏!
虽然现在很多人喜欢刷手机,但是对于长时间需要使用手机办公的人来说,手机屏幕还是太小了,当人一天二十四小时中要花费近十个小时摆弄手机,就会渴望手机屏幕能够大一点,至少看的时候,眼睛舒服一点。 因为嫌弃…...

芯盾时代数据安全产品体系,筑牢数据安全防线
芯盾时代数据安全治理(DSG)框架,以国家法律法规、行业监管标准、行业最佳实践为依据,从数据安全战略出发,以数据分类分级为支撑,构数据安全管理体系、数据安全技术体系、数据安全运营体系与数据安全监督评价…...

异位妊娠唯一相关的是年龄(U型曲线)
异位妊娠唯一相关的是年龄(U型曲线) 简介 异位妊娠,俗称宫外孕,是指受精卵在子宫体腔以外着床发育的异常妊娠过程 。正常情况下,受精卵会在子宫内着床并发育成胎儿,但在异位妊娠中,受精卵却在…...
CTF-WEB: 利用Web消息造成DOM XSS
如果索引中有类似如下代码 <!-- Ads to be inserted here --> <div idads> </div> <script>window.addEventListener(message, function(e) {document.getElementById(ads).innerHTML e.data;}); </script>这行代码的作用是将接收到的消息内容…...

【通俗易懂说模型】一篇弄懂几个经典CNN图像模型(AlexNet、VGGNet、ResNet)
🌈 个人主页:十二月的猫-CSDN博客 🔥 系列专栏: 🏀深度学习_十二月的猫的博客-CSDN博客 💪🏻 十二月的寒冬阻挡不了春天的脚步,十二点的黑夜遮蔽不住黎明的曙光 目录 1. 前言 2. …...
Unity世界坐标转成UI坐标
Unity世界坐标转成UI坐标 介绍转换代码合并方法总结 介绍 在Unity中官方提供了很多坐标转换的API,但是还没有一个API是将世界坐标系转换成UI的坐标系,世界坐标系在屏幕中的位置是不固定的所以有时候需要转换成UI坐标系**(注意这里不是转换成…...
自制游戏——斗罗大陆
很简陋,没有图,请见谅 // mine[0] 级数 // mine[1] 战力 //mine[2] 1 白虎 //mine[2] 2 昊天锤 //mine[2] 3 蓝银草 #include <bits/stdc.h> using namespace std; int mine[100],live3, dou 1, luo 1, da 1, bag[1000], huan 0, lia…...

MindStudio制作MindSpore TBE算子(四)算子测试(ST测试-Ascend910B/ModelArts)--失败尝试
上一节,MindStudio制作MindSpore TBE算子(三)算子测试(ST测试),因此缺乏对应的硬件环境导致无法进行ST测试,导致难以自安,今天搞来Ascend910B服务器来填坑,看看是否是硬件…...

二、交换机的vlan子设备接入
一、交换机的vlan设置-CSDN博客 二、交换机的vlan子设备接入-CSDN博客 接上篇的文章,本文接入了子设备 网络结构如下: 用路由器A和POE交换机B代替第一篇中的笔记本电脑,路由器A和交换机B都关闭DHCP服务,并分别接入一个IPC&#…...
EFCore相关知识分享
EFCore相关知识分享 文章目录 EFCore相关知识分享前言EFCore 的优势SaveChang的作用EF Core 查询优化小技巧使用投影(投影到所需的数据类型)延迟加载(Lazy Loading)与显式加载(Eager Loading)使用 AsNoTrac…...

检测网络安全漏洞 工具 网络安全 漏洞扫描 实验
实验一的名称为信息收集和漏洞扫描 实验环境:VMware下的kali linux2021和Windows7 32,网络设置均为NAT,这样子两台机器就在一个网络下。攻击的机器为kali,被攻击的机器为Windows 7。 理论知识记录: 1.信息收集的步骤 2.ping命令…...

deepseek + kimi 高效生成PPT
1.在deepseek中生成ppt大纲 2.将大纲复制到kimi中生成PPT kimi:https://kimi.moonshot.cn/...
JavaWeb学习-Mybatis(增删改查)
(一)Mybatis入门程序 1.创建springboot工程,并导入 mybatis的起步依赖、mysql的驱动包。(项目工程创建完成后,自动在pom.xml文件中,导入Mybatis依赖和MySQL驱动依赖) <dependencies> <!-- mybatis起步依赖 --> <dependency> …...

软考高项(二十四)法律法规和标准规范 ★重点集萃★
👑 个人主页 👑 :😜😜😜Fish_Vast😜😜😜 🐝 个人格言 🐝 :🧐🧐🧐说到做到,言出必行&am…...

龙虎榜——20250610
上证指数放量收阴线,个股多数下跌,盘中受消息影响大幅波动。 深证指数放量收阴线形成顶分型,指数短线有调整的需求,大概需要一两天。 2025年6月10日龙虎榜行业方向分析 1. 金融科技 代表标的:御银股份、雄帝科技 驱动…...

Linux 文件类型,目录与路径,文件与目录管理
文件类型 后面的字符表示文件类型标志 普通文件:-(纯文本文件,二进制文件,数据格式文件) 如文本文件、图片、程序文件等。 目录文件:d(directory) 用来存放其他文件或子目录。 设备…...

label-studio的使用教程(导入本地路径)
文章目录 1. 准备环境2. 脚本启动2.1 Windows2.2 Linux 3. 安装label-studio机器学习后端3.1 pip安装(推荐)3.2 GitHub仓库安装 4. 后端配置4.1 yolo环境4.2 引入后端模型4.3 修改脚本4.4 启动后端 5. 标注工程5.1 创建工程5.2 配置图片路径5.3 配置工程类型标签5.4 配置模型5.…...

python打卡day49
知识点回顾: 通道注意力模块复习空间注意力模块CBAM的定义 作业:尝试对今天的模型检查参数数目,并用tensorboard查看训练过程 import torch import torch.nn as nn# 定义通道注意力 class ChannelAttention(nn.Module):def __init__(self,…...

centos 7 部署awstats 网站访问检测
一、基础环境准备(两种安装方式都要做) bash # 安装必要依赖 yum install -y httpd perl mod_perl perl-Time-HiRes perl-DateTime systemctl enable httpd # 设置 Apache 开机自启 systemctl start httpd # 启动 Apache二、安装 AWStats࿰…...
大学生职业发展与就业创业指导教学评价
这里是引用 作为软工2203/2204班的学生,我们非常感谢您在《大学生职业发展与就业创业指导》课程中的悉心教导。这门课程对我们即将面临实习和就业的工科学生来说至关重要,而您认真负责的教学态度,让课程的每一部分都充满了实用价值。 尤其让我…...

C# 求圆面积的程序(Program to find area of a circle)
给定半径r,求圆的面积。圆的面积应精确到小数点后5位。 例子: 输入:r 5 输出:78.53982 解释:由于面积 PI * r * r 3.14159265358979323846 * 5 * 5 78.53982,因为我们只保留小数点后 5 位数字。 输…...

【MATLAB代码】基于最大相关熵准则(MCC)的三维鲁棒卡尔曼滤波算法(MCC-KF),附源代码|订阅专栏后可直接查看
文章所述的代码实现了基于最大相关熵准则(MCC)的三维鲁棒卡尔曼滤波算法(MCC-KF),针对传感器观测数据中存在的脉冲型异常噪声问题,通过非线性加权机制提升滤波器的抗干扰能力。代码通过对比传统KF与MCC-KF在含异常值场景下的表现,验证了后者在状态估计鲁棒性方面的显著优…...

零知开源——STM32F103RBT6驱动 ICM20948 九轴传感器及 vofa + 上位机可视化教程
STM32F1 本教程使用零知标准板(STM32F103RBT6)通过I2C驱动ICM20948九轴传感器,实现姿态解算,并通过串口将数据实时发送至VOFA上位机进行3D可视化。代码基于开源库修改优化,适合嵌入式及物联网开发者。在基础驱动上新增…...

Redis上篇--知识点总结
Redis上篇–解析 本文大部分知识整理自网上,在正文结束后都会附上参考地址。如果想要深入或者详细学习可以通过文末链接跳转学习。 1. 基本介绍 Redis 是一个开源的、高性能的 内存键值数据库,Redis 的键值对中的 key 就是字符串对象,而 val…...