0

0

如何在Golang中利用Bun框架操作数据库 Go语言轻量级ORM选择

P粉602998670

P粉602998670

发布时间:2026-03-13 13:22:03

|

771人浏览过

|

来源于php中文网

原创

Bun 是 SQL 构建器 + 运行时映射器,非 ORM:不生成 SQL、不维护状态、无懒加载;需手动写查询逻辑,显式调用 Scan/Exec,结构体须标注 pk/array 标签,JSONB 用 json.RawMessage,事务需正确传递 tx 实例,分页推荐游标而非 OFFSET/LIMIT。

如何在golang中利用bun框架操作数据库 go语言轻量级orm选择

为什么 Bun 不是“ORM”而是“SQL 构建器 + 运行时映射器”

Bun 的定位常被误读为 ORM,但它实际不生成 SQL、不维护对象状态、也不做懒加载或关联预加载的自动推导。它更像一个带结构体映射能力的 bun.DB + QueryBuilder 组合。这意味着你得自己写 SELECTJOINWHERE 逻辑,Bun 只负责把结果塞进 struct,或把 struct 翻译成参数化 SQL。

常见错误现象:db.NewSelect().Model(&users).Where("age > ?", 18) 看似简单,但如果你忘了加 .Scan(ctx),它根本不会执行;或者用 db.Model(&u).Set("name = ?", "x").WherePK() 更新时漏了主键字段标签,就会更新全表。

  • 所有查询必须显式调用 .Scan().Exec().Count() 才真正触发数据库操作
  • Model() 的参数必须是指针(&user),否则 Scan() 无法写入
  • Bun 不自动识别 ID 字段为主键——必须用 sql:",pk" 标签声明
  • 嵌套 struct 关联需手动 .Relation("Profile"),且被关联 struct 必须有外键字段(如 ProfileID int)并打上对应标签

怎么让 Bun 正确映射 PostgreSQL 的 JSONB 和数组字段

PostgreSQL 的 jsonbtext[] 在 Go 中没有默认对应类型,Bun 不会自动转换。直接用 stringjsonb 会导致扫描失败,报错类似 can't scan into dest: cannot assign <nil> to *string</nil>(尤其当字段允许 NULL 时)。

使用场景:API 响应元数据存为 jsonb,权限列表存在 text[] 中。

立即学习go语言免费学习笔记(深入)”;

Nanonets
Nanonets

基于AI的自学习OCR文档处理,自动捕获文档数据

下载
  • jsonb 字段推荐用 json.RawMessage 类型,避免提前解析开销;若需结构化访问,定义具体 struct 并实现 driver.Valuer + sql.Scanner
  • text[] 对应 []string,但必须加 pg:",array" 标签(Bun 用的是 pg 驱动,不是 database/sql 原生数组支持)
  • 别用 interface{} 接 jsonb——Bun 无法推断目标类型,会 panic
  • NULL 值处理:字段类型用指针(如 *json.RawMessage*[]string),否则 NULL 扫描时报错

事务里用 Bun 更新失败却没报错?检查 Context 和 ErrGroup 配合方式

Bun 的 db.InTx() 返回新 DB 实例,所有操作必须用这个实例,而不是原始 db。更隐蔽的问题是:在 errgroup.Group 里并发调用 tx.Model().Update(),如果某个 goroutine 拿到的 tx 是闭包外的变量,可能实际用的是同一个 tx 实例,而 Bun 的事务内部不是 goroutine-safe 的——会导致竞态或静默失败。

  • 事务内所有 DB 操作必须用传入回调函数里的 tx,不能复用外部 db
  • 不要在 errgroup.Go() 回调里直接捕获 tx 变量;要么显式传参,要么用 func(tx *bun.DB) error 包一层
  • 事务中任何一步出错(比如约束冲突),Bun 不自动 rollback——必须靠 return err 触发 InTx() 的回滚逻辑
  • 注意 context 超时:若 InTx() 传入的 ctx 已 cancel,后续 tx.Model().Insert() 会立即返回 context canceled,但不会提示“事务已失效”

用 Bun 做分页时 OFFSET LIMIT 性能差?改用游标分页 + WHERE

Bun 默认分页靠 .Offset().Limit(),在千万级表上 OFFSET 1000000 会拖垮查询。这不是 Bun 的问题,而是 PostgreSQL 自身限制。但很多人卡在“Bun 没提供游标分页 DSL”,就硬扛着用 offset。

实际上 Bun 完全支持手写游标逻辑:用上一页最后一条记录的排序字段(如 idcreated_at)构造 WHERE id > ? 条件,再配 LIMIT

  • 不要依赖 bun.Paginator——它只是封装 offset/limit,不解决底层性能问题
  • 游标字段必须有索引(CREATE INDEX ON users (created_at, id)),且排序方向一致(ORDER BY created_at DESC, id DESC
  • 游标值要用上一页最后一条的原始值(不是字符串拼接后的),避免类型转换误差
  • 首次请求无游标时,用 WHERE 1=1 占位,保持 SQL 结构统一,别动态拼 SQL

复杂点在于游标需要业务语义对齐:比如按时间分页时,同一秒可能有多条记录,必须用第二排序字段(如 id)打破平局,否则漏数据或重复。这点容易被忽略,一上线就出错。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

211

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

247

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

356

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

214

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

409

2024.05.21

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

490

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

201

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

1479

2025.06.17

Python异步编程与Asyncio高并发应用实践
Python异步编程与Asyncio高并发应用实践

本专题围绕 Python 异步编程模型展开,深入讲解 Asyncio 框架的核心原理与应用实践。内容包括事件循环机制、协程任务调度、异步 IO 处理以及并发任务管理策略。通过构建高并发网络请求与异步数据处理案例,帮助开发者掌握 Python 在高并发场景中的高效开发方法,并提升系统资源利用率与整体运行性能。

37

2026.03.12

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 6.2万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号