Tio Boot DocsTio Boot Docs
Home
  • java-db
  • api-table
  • mysql
  • postgresql
  • oceanbase
  • Enjoy
  • Tio Boot Admin
  • ai_agent
  • translator
  • knowlege_base
  • ai-search
  • 案例
Abount
  • Github
  • Gitee
Home
  • java-db
  • api-table
  • mysql
  • postgresql
  • oceanbase
  • Enjoy
  • Tio Boot Admin
  • ai_agent
  • translator
  • knowlege_base
  • ai-search
  • 案例
Abount
  • Github
  • Gitee
  • 01_tio-boot 简介

    • tio-boot:新一代高性能 Java Web 开发框架
    • tio-boot 入门示例
    • Tio-Boot 配置 : 现代化的配置方案
    • tio-boot 整合 Logback
    • tio-boot 整合 hotswap-classloader 实现热加载
    • 自行编译 tio-boot
    • 最新版本
    • 开发规范
  • 02_部署

    • 使用 Maven Profile 实现分环境打包 tio-boot 项目
    • Maven 项目配置详解:依赖与 Profiles 配置
    • tio-boot 打包成 FatJar
    • 使用 GraalVM 构建 tio-boot Native 程序
    • 使用 Docker 部署 tio-boot
    • 部署到 Fly.io
    • 部署到 AWS Lambda
    • 到阿里云云函数
    • 使用 Deploy 工具部署
    • 使用Systemctl启动项目
    • 使用 Jenkins 部署 Tio-Boot 项目
    • 使用 Nginx 反向代理 Tio-Boot
    • 使用 Supervisor 管理 Java 应用
    • 已过时
    • 胖包与瘦包的打包与部署
  • 03_配置

    • 配置参数
    • 服务器监听器
    • 内置缓存系统 AbsCache
    • 使用 Redis 作为内部 Cache
    • 静态文件处理器
    • 基于域名的静态资源隔离
    • DecodeExceptionHandler
    • 开启虚拟线程(Virtual Thread)
    • 框架级错误通知
  • 04_原理

    • 生命周期
    • 请求处理流程
    • 重要的类
  • 05_json

    • Json
    • 接受 JSON 和响应 JSON
    • 响应实体类
  • 06_web

    • 概述
    • 接收请求参数
    • 接收日期参数
    • 接收数组参数
    • 返回字符串
    • 返回文本数据
    • 返回网页
    • 请求和响应字节
    • 文件上传
    • 文件下载
    • 返回视频文件并支持断点续传
    • http Session
    • Cookie
    • HttpRequest
    • HttpResponse
    • Resps
    • RespBodyVo
    • Controller拦截器
    • 请求拦截器
    • LoggingInterceptor
    • 全局异常处理器
    • 异步处理
    • 动态 返回 CSS 实现
    • 返回图片
    • 跨域
    • 添加 Controller
    • Transfer-Encoding: chunked 实时音频播放
    • Server-Sent Events (SSE)
    • handler入门
    • 返回 multipart
    • 待定
    • 自定义 Handler 转发请求
    • 使用 HttpForwardHandler 转发所有请求
    • 常用工具类
    • HTTP Basic 认证
    • Http响应加密
    • 使用零拷贝发送大文件
    • 分片上传
    • 接口访问统计
    • 接口请求和响应数据记录
    • WebJars
    • JProtobuf
    • 测速
    • Gzip Bomb:使用压缩炸弹防御恶意爬虫
  • 07_validate

    • 数据紧校验规范
    • 参数校验
  • 08_websocket

    • 使用 tio-boot 搭建 WebSocket 服务
    • WebSocket 聊天室项目示例
  • 09_java-db

    • java‑db
    • 操作数据库入门示例
    • SQL 模板 (SqlTemplates)
    • 数据源配置与使用
    • ActiveRecord
    • Db 工具类
    • 批量操作
    • Model
    • Model生成器
    • 注解
    • 异常处理
    • 数据库事务处理
    • Cache 缓存
    • Dialect 多数据库支持
    • 表关联操作
    • 复合主键
    • Oracle 支持
    • Enjoy SQL 模板
    • 整合 Enjoy 模板最佳实践
    • 多数据源支持
    • 独立使用 ActiveRecord
    • 调用存储过程
    • java-db 整合 Guava 的 Striped 锁优化
    • 生成 SQL
    • 通过实体类操作数据库
    • java-db 读写分离
    • Spring Boot 整合 Java-DB
    • like 查询
    • 常用操作示例
    • Druid 监控集成指南
    • SQL 统计
  • 10_api-table

    • ApiTable 概述
    • 使用 ApiTable 连接 SQLite
    • 使用 ApiTable 连接 Mysql
    • 使用 ApiTable 连接 Postgres
    • 使用 ApiTable 连接 TDEngine
    • 使用 api-table 连接 oracle
    • 使用 api-table 连接 mysql and tdengine 多数据源
    • EasyExcel 导出
    • EasyExcel 导入
    • 预留
    • 预留
    • ApiTable 实现增删改查
    • 数组类型
    • 单独使用 ApiTable
    • TQL(Table SQL)前端输入规范
  • 11_aop

    • JFinal-aop
    • Aop 工具类
    • 配置
    • 配置
    • 独立使用 JFinal Aop
    • @AImport
    • 自定义注解拦截器
    • 原理解析
  • 12_cache

    • Caffine
    • Jedis-redis
    • hutool RedisDS
    • Redisson
    • Caffeine and redis
    • CacheUtils 工具类
    • 使用 CacheUtils 整合 caffeine 和 redis 实现的两级缓存
    • 使用 java-db 整合 ehcache
    • 使用 java-db 整合 redis
    • Java DB Redis 相关 Api
    • redis 使用示例
  • 13_认证和权限

    • FixedTokenInterceptor
    • TokenManager
    • 数据表
    • 匿名登录
    • 注册和登录
    • 个人中心
    • 重置密码
    • Google 登录
    • 短信登录
    • 移动端微信登录
    • 移动端重置密码
    • 微信登录
    • 移动端微信登录
    • 权限校验注解
    • Sa-Token
    • sa-token 登录注册
    • StpUtil.isLogin() 源码解析
  • 14_i18n

    • i18n
  • 15_enjoy

    • tio-boot 整合 Enjoy 模版引擎文档
    • Tio-Boot 整合 Java-DB 与 Enjoy 模板引擎示例
    • 引擎配置
    • 表达式
    • 指令
    • 注释
    • 原样输出
    • Shared Method 扩展
    • Shared Object 扩展
    • Extension Method 扩展
    • Spring boot 整合
    • 独立使用 Enjoy
    • tio-boot enjoy 自定义指令 localeDate
    • PromptEngine
    • Enjoy 入门示例-擎渲染大模型请求体
    • Tio Boot + Enjoy:分页与 SEO 实战指南
    • Tio Boot + Enjoy:分页与 SEO 实战指南
    • Tio Boot + Enjoy:分页与 SEO 实战指南
  • 16_定时任务

    • Quartz 定时任务集成指南
    • 分布式定时任务 xxl-jb
    • cron4j 使用指南
  • 17_tests

    • TioBootTest 类
  • 18_tio

    • TioBootServer
    • 独立端口启动 TCP 服务器
    • 内置 TCP 处理器
    • 独立启动 UDPServer
    • 使用内置 UDPServer
    • t-io 消息处理流程
    • tio-运行原理详解
    • TioConfig
    • ChannelContext
    • Tio 工具类
    • 业务数据绑定
    • 业务数据解绑
    • 发送数据
    • 关闭连接
    • Packet
    • 监控: 心跳
    • 监控: 客户端的流量数据
    • 监控: 单条 TCP 连接的流量数据
    • 监控: 端口的流量数据
    • 单条通道统计: ChannelStat
    • 所有通道统计: GroupStat
    • 资源共享
    • 成员排序
    • SSL
    • DecodeRunnable
    • 使用 AsynchronousSocketChannel 响应数据
    • 拉黑 IP
    • 深入解析 Tio 源码:构建高性能 Java 网络应用
  • 19_aio

    • ByteBuffer
    • AIO HTTP 服务器
    • 自定义和线程池和池化 ByteBuffer
    • AioHttpServer 应用示例 IP 属地查询
    • 手写 AIO Http 服务器
  • 20_netty

    • Netty TCP Server
    • Netty Web Socket Server
    • 使用 protoc 生成 Java 包文件
    • Netty WebSocket Server 二进制数据传输
    • Netty 组件详解
  • 21_netty-boot

    • Netty-Boot
    • 原理解析
    • 整合 Hot Reload
    • 整合 数据库
    • 整合 Redis
    • 整合 Elasticsearch
    • 整合 Dubbo
    • Listener
    • 文件上传
    • 拦截器
    • Spring Boot 整合 Netty-Boot
    • SSL 配置指南
    • ChannelInitializer
    • Reserve
  • 22_MQ

    • Mica-mqtt
    • EMQX
    • Disruptor
  • 23_tio-utils

    • tio-utils
    • HttpUtils
    • Notification
    • Email
    • JSON
    • File
    • Base64
    • 上传和下载
    • Http
    • Telegram
    • RsaUtils
    • EnvUtils 配置工具
    • 系统监控
    • 线程
    • 虚拟线程
    • 毫秒并发 ID (MCID) 生成方案
  • 24_tio-http-server

    • 使用 Tio-Http-Server 搭建简单的 HTTP 服务
    • tio-boot 添加 HttpRequestHandler
    • 在 Android 上使用 tio-boot 运行 HTTP 服务
    • tio-http-server-native
    • handler 常用操作
  • 25_tio-websocket

    • WebSocket 服务器
    • WebSocket Client
    • TCP数据转发
  • 26_tio-im

    • 通讯协议文档
    • ChatPacket.proto 文档
    • java protobuf
    • 数据表设计
    • 创建工程
    • 登录
    • 历史消息
    • 发消息
  • 27_mybatis

    • Tio-Boot 整合 MyBatis
    • 使用配置类方式整合 MyBatis
    • 整合数据源
    • 使用 mybatis-plus 整合 tdengine
    • 整合 mybatis-plus
  • 28_mongodb

    • tio-boot 使用 mongo-java-driver 操作 mongodb
  • 29_elastic-search

    • Elasticsearch
    • JavaDB 整合 ElasticSearch
    • Elastic 工具类使用指南
    • Elastic-search 注意事项
    • ES 课程示例文档
  • 30_magic-script

    • tio-boot 与 magic-script 集成指南
  • 31_groovy

    • tio-boot 整合 Groovy
  • 32_firebase

    • 整合 google firebase
    • Firebase Storage
    • Firebase Authentication
    • 使用 Firebase Admin SDK 进行匿名用户管理与自定义状态标记
    • 导出用户
    • 注册回调
    • 登录注册
  • 33_文件存储

    • 文件上传数据表
    • 本地存储
    • 存储文件到 亚马逊 S3
    • Cloudflare R2
    • 存储文件到 腾讯 COS
    • 存储文件到 阿里云 OSS
  • 34_spider

    • jsoup
    • 爬取 z-lib.io 数据
    • 整合 WebMagic
    • WebMagic 示例:爬取学校课程数据
    • Playwright
    • Flexmark (Markdown 处理器)
    • tio-boot 整合 Playwright
    • 缓存网页数据
  • 36_integration_thirty_party

    • 整合 okhttp
    • 整合 GrpahQL
    • 集成 Mailjet
    • 整合 ip2region
    • 整合 GeoLite 离线库
    • 整合 Lark 机器人指南
    • 集成 Lark Mail 实现邮件发送
    • Thymeleaf
    • Swagger
    • Clerk 验证
  • 37_dubbo

    • 概述
    • dubbo 2.6.0
    • dubbo 2.6.0 调用过程
    • dubbo 3.2.0
  • 38_spring

    • Spring Boot Web 整合 Tio Boot
    • spring-boot-starter-webflux 整合 tio-boot
    • tio-boot 整合 spring-boot-starter
    • Tio Boot 整合 Spring Boot Starter db
    • Tio Boot 整合 Spring Boot Starter Data Redis 指南
  • 39_spring-cloud

    • tio-boot spring-cloud
  • 40_quarkus

    • Quarkus(无 HTTP)整合 tio-boot(有 HTTP)
    • tio-boot + Quarkus + Hibernate ORM Panache
  • 41_postgresql

    • PostgreSQL 安装
    • PostgreSQL 主键自增
    • PostgreSQL 日期类型
    • Postgresql 金融类型
    • PostgreSQL 数组类型
    • 索引
    • PostgreSQL 查询优化
    • 获取字段类型
    • PostgreSQL 全文检索
    • PostgreSQL 向量
    • PostgreSQL 优化向量查询
    • PostgreSQL 其他
  • 42_mysql

    • 使用 Docker 运行 MySQL
    • 常见问题
  • 43_oceanbase

    • 快速体验 OceanBase 社区版
    • 快速上手 OceanBase 数据库单机部署与管理
    • 诊断集群性能
    • 优化 SQL 性能指南
    • 待定
  • 49_jooq

    • 使用配置类方式整合 jOOQ
    • tio-boot + jOOQ 事务管理
    • 批量操作与性能优化
    • 整合agroal
    • 代码生成与类型安全
    • 基于 Record / POJO 增删改查
    • UPSERT、批量更新、返回主键与高级 SQL
    • 的多表关联查询、DTO 投影、聚合统计与视图封装
    • 的窗口函数、CTE、JSON 查询与 PostgreSQL 高级 SQL 实战
    • tio-boot + jOOQ 的审计字段、乐观锁、数据权限与企业级 Repository 设计
    • 测试策略、SQL 日志、性能诊断与生产排障
    • 多租户、读写分离与多数据源设计
    • 代码生成治理、数据库迁移与团队协作规范实战
  • 50_media

    • JAVE 提取视频中的声音
    • Jave 提取视频中的图片
    • 待定
  • 51_asr

    • Whisper-JNI
  • 54_native-media

    • java-native-media
    • JNI 入门示例
    • mp3 拆分
    • mp4 转 mp3
    • 使用 libmp3lame 实现高质量 MP3 编码
    • Linux 编译
    • macOS 编译
    • 从 JAR 包中加载本地库文件
    • 支持的音频和视频格式
    • 任意格式转为 mp3
    • 通用格式转换
    • 通用格式拆分
    • 视频合并
    • VideoToHLS
    • split_video_to_hls 支持其他语言
    • 持久化 HLS 会话
    • 获取视频长度
    • 保存视频的最后一帧
    • 添加水印
    • linux版本
  • 55_cv

    • 使用 Java 运行 YOLOv8 ONNX 模型进行目标检测
    • tio-boot整合yolo
    • ONNX Runtime 推理说明
  • 58_telegram4j

    • 数据库设计
    • 基于 HTTP 协议开发 Telegram 翻译机器人
    • 基于 MTProto 协议开发 Telegram 翻译机器人
    • 过滤旧消息
    • 保存机器人消息
    • 定时推送
    • 增加命令菜单
    • 使用 telegram-Client
    • 使用自定义 StoreLayout
    • 延迟测试
    • Reactor 错误处理
    • Telegram4J 常见错误处理指南
  • 59_telegram-bots

    • TelegramBots 入门指南
    • 使用工具库 telegram-bot-base 开发翻译机器人
  • 60_LLM

    • 简介
    • 流式生成
    • 图片多模态输入
    • 协议自动转换 Google Gemini示例
    • 请求记录
    • 限流和错误处理
    • 整合Gemini realtime模型
    • Voice Agent 前端接入接口文档
    • 整合千问realtime模型
    • 增强检索(RAG)
    • 搜索+AI
    • AI 问答
    • 连接代码执行器
  • 61_ai_agent

    • 数据库设计
    • 示例问题管理
    • 会话管理
    • 历史记录
    • Perplexity API
    • 意图识别
    • 智能问答
    • 文件上传与解析文档
    • 翻译
    • 名人搜索功能实现
    • Ai studio gemini youbue 问答使用说明
    • 自建 YouTube 字幕问答系统
    • 自建 获取 youtube 字幕服务
    • 使用 OpenAI ASR 实现语音识别接口(Java 后端示例)
    • 定向搜索
    • 16
    • 17
    • 18
    • 在 tio-boot 应用中整合 ai-agent
    • 16
  • 63_knowlege_base

    • 数据库设计
    • 用户登录实现
    • 模型管理
    • 知识库管理
    • 文档拆分
    • 片段向量
    • 命中测试
    • 文档管理
    • 片段管理
    • 问题管理
    • 应用管理
    • 向量检索
    • 推理问答
    • 问答模块
    • 统计分析
    • 用户管理
    • api 管理
    • 存储文件到 S3
    • 文档解析优化
    • 片段汇总
    • 段落分块与检索
    • 多文档解析
    • 对话日志
    • 检索性能优化
    • Milvus
    • 文档解析方案和费用对比
    • 离线运行向量模型
  • 64_ai-search

    • ai-search 项目简介
    • ai-search 数据库文档
    • ai-search SearxNG 搜索引擎
    • ai-search Jina Reader API
    • ai-search Jina Search API
    • ai-search 搜索、重排与读取内容
    • ai-search PDF 文件处理
    • ai-search 推理问答
    • Google Custom Search JSON API
    • ai-search 意图识别
    • ai-search 问题重写
    • ai-search 系统 API 接口 WebSocket 版本
    • ai-search 搜索代码实现 WebSocket 版本
    • ai-search 生成建议问
    • ai-search 生成问题标题
    • ai-search 历史记录
    • Discover API
    • 翻译
    • Tavily Search API 文档
    • 对接 Tavily Search
    • 火山引擎 DeepSeek
    • 对接 火山引擎 DeepSeek
    • ai-search 搜索代码实现 SSE 版本
    • jar 包部署
    • Docker 部署
    • 爬取一个静态网站的所有数据
    • 网页数据预处理
    • 网页数据检索与问答流程整合
  • 65_ai-coding

    • Cline 提示词
    • Cline 提示词-中文版本
  • 66_java-uni-ai-server

    • 语音合成系统
    • Fish.audio TTS 接口说明文档与 Java 客户端封装
    • 整合 fishaudio 到 java-uni-ai-server 项目
    • 待定
  • 67_java-llm-proxy

    • 使用tio-boot搭建多模型LLM代理服务
  • 68_java-kit-server

    • Java 执行 python 代码
    • 通过大模型执行 Python 代码
    • 执行 Python (Manim) 代码
    • 待定
    • 待定
    • 待定
    • 视频下载增加水印说明文档
  • 69_ai-brower

    • AI Browser:基于用户指令的浏览器自动化系统
    • 提示词
    • dom构建- buildDomTree.js
    • dom构建- 将网页可点击元素提取与可视化
    • 提取网内容
    • 启动浏览器
    • 操作浏览器指令
  • 70_tio-boot-admin

    • 入门指南
    • 初始化数据
    • token 存储
    • 与前端集成
    • 文件上传
    • 网络请求
    • 多图片管理
    • 单图片管理(只读模式)
    • 布尔值管理
    • 字段联动
    • Word 管理
    • PDF 管理
    • 文章管理
    • 富文本编辑器
  • 73_tio-mail-wing

    • tio-mail-wing简介
    • 任务1:实现POP3系统
    • 使用 getmail 验证 tio-mail-wing POP3 服务
    • 任务2:实现 SMTP 服务
    • 数据库初始化文档
    • 用户管理
    • 邮件管理
    • 任务3:实现 SMTP 服务 数据库版本
    • 任务4:实现 POP3 服务(数据库版本)
    • IMAP 协议
    • 拉取多封邮件
    • 任务5:实现 IMAP 服务(数据库版本)
    • IMAP实现讲解
    • IMAP 手动测试脚本
    • IMAP 认证机制
    • 主动推送
  • 74_tio-mcp-server

    • 实现 MCP Server 开发指南
  • 75_tio-sip

    • SIP Server 第一版原理说明
    • SIP Server 第一版实战
    • 一、Windows 平台测试
    • SIP Server 第二版实战
    • SIP Server 第三版实战
    • 性能优化
    • 基于 MediaProcessor 对接 Realtime 模型说明
    • 对接大语言模型
    • 支持 G722 宽带语音
    • G722编码和解码
    • 会话级采样率转换
    • /zh/75_tio-sip/12.html
    • 增加 9196 回声测试分机
    • 语音系统链路说明
    • 一、Gemini Realtime 的打断机制
  • 76_manim

    • Teach me anything - 基于大语言的知识点讲解视频生成系统
    • Manim 开发环境搭建
    • 生成场景提示词
    • 生成代码
    • 完整脚本示例
    • TTS服务端
    • 废弃
    • 废弃
    • 废弃
    • 使用 SSE 流式传输生成进度的实现文档
    • 整合全流程完整文档
    • HLS 动态推流技术文档
    • manim 分场景生成代码
    • 分场景运行代码及流式播放支持
    • 分场景业务端完整实现流程
    • Maiim布局管理器
    • 仅仅生成场景代码
    • 使用 modal 运行 manim 代码
    • Python 使用 Modal GPU 加速渲染
    • Modal 平台 GPU 环境下运行 Manim
    • Modal Manim OpenGL 安装与使用
    • 优化 GPU 加速
    • 生成视频封面流程
    • Java 调用 manim 命令 执行代码 生成封面
    • Manim 图像生成服务客户端文档
    • manim render help
    • 显示 中文公式
    • ManimGL(manimgl)
    • Manim 实战入门:用代码创造数学动画
    • 欢迎
  • 80_性能测试

    • 压力测试 - tio-http-serer
    • 压力测试 - tio-boot
    • 压力测试 - tio-boot-native
    • 压力测试 - netty-boot
    • 性能测试对比
    • TechEmpower FrameworkBenchmarks
    • 压力测试 - tio-boot 12 C 32G
    • HTTP/1.1 Pipelining 性能测试报告
    • tio-boot vs Quarkus 性能对比测试报告
  • 81_tio-boot

    • 简介
    • Swagger 整合到 Tio-Boot 中的指南
    • 待定
    • 待定
    • 高性能网络编程中的 ByteBuffer 分配与回收策略
    • TioBootServerHandler 源码解析
  • 99_案例

    • 封装 IP 查询服务
    • tio-boot 案例 - 全局异常捕获与企业微信群通知
    • tio-boot 案例 - 文件上传和下载
    • tio-boot 案例 - 整合 ant design pro 增删改查
    • tio-boot 案例 - 流失响应
    • tio-boot 案例 - 增强检索
    • tio-boot 案例 - 整合 function call
    • tio-boot 案例 - 定时任务 监控 PostgreSQL、Redis 和 Elasticsearch
    • Tio-Boot 案例:使用 SQLite 整合到登录注册系统
    • tio-boot 案例 - 执行 shell 命令

UPSERT、批量更新、返回主键与高级 SQL

  • 一、为什么这一篇很重要
  • 二、先明确几种写入语义
    • 1. insert
    • 2. update
    • 3. 应用层 saveOrUpdate
    • 4. 数据库层 UPSERT
  • 三、PostgreSQL UPSERT 基础概念
  • 四、jOOQ 中的 UPSERT 写法
  • 4.1 基于唯一键的 UPSERT
    • 说明
  • 4.2 使用 excluded(...)
  • 4.3 基于主键的 UPSERT
  • 五、UPSERT 返回主键或完整对象
  • 5.1 插入后返回主键
    • 说明
  • 5.2 插入后返回整条 Record
    • 适用场景
  • 5.3 插入后返回 POJO
  • 5.4 UPSERT 后返回结果
  • 六、returning 和 Record.insert() 回填主键的区别
    • 1. record.insert() 后取主键
    • 2. returning(...)
  • 七、批量更新怎么做
    • 1. 相同 SQL 模板,多组参数
    • 2. 一条 SQL,更新多行
    • 3. 更复杂的批量差异更新
  • 八、同条件批量更新
    • 8.1 一组 id 统一更新
    • 特点
  • 九、差异化批量更新:JDBC Batch 风格
  • 9.1 批量提交多条 update
    • 说明
  • 9.2 基于 Record 的批量存储
    • 注意
  • 十、复杂差异更新:CASE WHEN
    • 特点
  • 十一、批量插入再深入一点
  • 11.1 用 batchInsert 批量导入 POJO
  • 11.2 分批导入
    • 为什么推荐分批
  • 十二、插入多行并返回结果
    • 1. 小批量逐条 insert + returning
    • 2. 构造多 values 插入 + returning
    • 适用场景
  • 十三、带返回值的 update / delete
  • 13.1 update 后返回更新结果
    • 适合场景
  • 13.2 delete 后返回被删除记录
  • 十四、高级 SQL:子查询更新
  • 十五、高级 SQL:存在性判断
    • 优点
  • 十六、高级 SQL:聚合查询
  • 十七、高级 SQL:表达式字段与别名
  • 十八、推荐的 DAO 示例
  • 十九、UPSERT 与事务边界的关系
  • 二十、什么时候优先用 UPSERT
    • 1. 存在明确唯一键
    • 2. 业务允许“有则改,无则增”
    • 3. 有并发写入可能
  • 二十一、什么时候不建议滥用 UPSERT
    • 1. 业务规则复杂
    • 2. 需要非常明确地区分“新增成功”和“更新成功”
  • 二十二、几个常见坑
    • 1. UPSERT 必须依赖唯一键或主键
    • 2. 批量更新不一定要用 batch
    • 3. returning() 很强,但不要无脑全返回
    • 4. batchStore() 虽方便,但意图不如显式 batch 清楚
    • 5. PostgreSQL 特性很强,但要承认它是方言能力
  • 二十三、总结

在前几篇中,我们已经完成了:

  • tio-boot + jOOQ 纯配置类整合
  • 事务管理
  • Agroal / Druid 数据源整合
  • jOOQ Codegen 强类型升级
  • 基于 Record / POJO 的增删改查、分页、动态条件、批量插入

到这里,已经可以覆盖大多数常规 CRUD 场景。

但在真实项目中,还会频繁遇到几类更“生产化”的问题:

  • 插入时如果已存在,如何自动改成更新
  • 批量更新怎么写才清晰
  • 插入后如何可靠拿到主键或完整返回值
  • PostgreSQL 的 returning、on conflict 如何用 jOOQ 表达
  • 如何把复杂 SQL 继续保持在“强类型 + 可维护”的轨道上

本文就围绕这些问题,系统讲透:

  • PostgreSQL UPSERT
  • 批量更新
  • 返回主键与 returning
  • 批量写入的几种方式
  • 常见高级 SQL 写法
  • 在 tio-boot + jOOQ 中的推荐工程实践

一、为什么这一篇很重要

前几篇主要解决的是“能不能优雅地做 CRUD”。

这一篇解决的是:

当项目进入真实业务阶段后,如何把数据库写操作做得更稳、更高效、更贴近 PostgreSQL 能力。

尤其是 PostgreSQL,有几个非常重要的能力:

  • insert ... returning
  • insert ... on conflict do update
  • 批量插入
  • 公共表达式与子查询
  • 强大的函数和表达式系统

而 jOOQ 的价值就在于:

把这些数据库原生能力,继续以 Java 强类型 DSL 的方式表达出来。

也就是说,jOOQ 并不会“抹平”数据库特性,反而会尽量保留数据库原生表达力。


二、先明确几种写入语义

在进入代码之前,先把几个容易混淆的概念区分开。


1. insert

语义很明确:

就是新增,如果冲突了就报错

适合场景:

  • 业务明确要求新增
  • 重复数据必须视为异常
  • 主键或唯一键冲突应立即暴露

2. update

语义也很明确:

只更新已有数据,不存在就更新 0 行

适合场景:

  • 明确按 id / 唯一键更新
  • 业务上“不存在就不更新”是合理行为

3. 应用层 saveOrUpdate

它的逻辑通常是:

  • 先查
  • 再决定 insert 或 update

例如:

if (id == null) {
  insert();
} else {
  update();
}

或者:

if (exist == null) {
  insert();
} else {
  update();
}

它的特点是:

  • 逻辑清晰
  • 容易理解
  • 但通常至少两步数据库交互
  • 并发下未必原子

4. 数据库层 UPSERT

这类语义在 PostgreSQL 中通常是:

insert into ...
on conflict (...) do update ...

它的特点是:

  • 一条 SQL 完成
  • 原子性更好
  • 更适合并发写入
  • 更贴近数据库原生能力

所以要记住一句话:

应用层 saveOrUpdate 和 PostgreSQL UPSERT,不是一回事。


三、PostgreSQL UPSERT 基础概念

PostgreSQL 的 UPSERT 依赖唯一约束或主键冲突检测。

例如,假设 system_admin.login_name 上有唯一约束:

ALTER TABLE system_admin
ADD CONSTRAINT uk_system_admin_login_name UNIQUE (login_name);

那么就可以写:

insert into system_admin(login_name, password)
values ('litong', '123456')
on conflict (login_name)
do update set password = excluded.password;

语义是:

  • 如果 login_name 不存在,执行插入
  • 如果 login_name 已存在,执行更新

这里的:

excluded.password

表示本次尝试插入的那一行里的值。


四、jOOQ 中的 UPSERT 写法

先假设我们已经有静态导入:

import static demo.jooq.gen.tables.SystemAdmin.SYSTEM_ADMIN;

4.1 基于唯一键的 UPSERT

public int upsertByLoginName(String loginName, String password) {
  return useDsl()
      .insertInto(SYSTEM_ADMIN)
      .set(SYSTEM_ADMIN.LOGIN_NAME, loginName)
      .set(SYSTEM_ADMIN.PASSWORD, password)
      .onConflict(SYSTEM_ADMIN.LOGIN_NAME)
      .doUpdate()
      .set(SYSTEM_ADMIN.PASSWORD, password)
      .execute();
}

说明

这一段对应的 PostgreSQL 语义就是:

insert into system_admin(login_name, password)
values (?, ?)
on conflict (login_name)
do update set password = ?

它的优点非常明显:

  • 一条 SQL 完成
  • 无需先查再写
  • 并发下更稳妥
  • 意图清晰

4.2 使用 excluded(...)

在 UPSERT 中,更新值往往直接来自“插入尝试值”。

这时可以使用 excluded(...) 风格表达。

public int upsertByLoginName2(String loginName, String password) {
  return useDsl()
      .insertInto(SYSTEM_ADMIN)
      .set(SYSTEM_ADMIN.LOGIN_NAME, loginName)
      .set(SYSTEM_ADMIN.PASSWORD, password)
      .onConflict(SYSTEM_ADMIN.LOGIN_NAME)
      .doUpdate()
      .set(SYSTEM_ADMIN.PASSWORD, DSL.excluded(SYSTEM_ADMIN.PASSWORD))
      .execute();
}

这个写法更贴近 PostgreSQL 原生 SQL:

  • 插入什么值
  • 冲突后就用这次插入值覆盖

这样做的好处是:后续字段多时更统一。


4.3 基于主键的 UPSERT

如果业务语义是“按主键冲突处理”,也可以这样写:

public int upsertById(Integer id, String loginName, String password) {
  return useDsl()
      .insertInto(SYSTEM_ADMIN)
      .set(SYSTEM_ADMIN.ID, id)
      .set(SYSTEM_ADMIN.LOGIN_NAME, loginName)
      .set(SYSTEM_ADMIN.PASSWORD, password)
      .onConflict(SYSTEM_ADMIN.ID)
      .doUpdate()
      .set(SYSTEM_ADMIN.LOGIN_NAME, DSL.excluded(SYSTEM_ADMIN.LOGIN_NAME))
      .set(SYSTEM_ADMIN.PASSWORD, DSL.excluded(SYSTEM_ADMIN.PASSWORD))
      .execute();
}

不过在自增主键场景里,业务上更常见的 UPSERT 键通常不是 id,而是某种业务唯一键。

例如:

  • login_name
  • email
  • tenant_id + code
  • biz_type + biz_id

五、UPSERT 返回主键或完整对象

在 PostgreSQL 中,returning 是非常强大的能力。

它允许你在写入后直接拿回:

  • 主键
  • 若干字段
  • 甚至整行

jOOQ 对此支持很好。


5.1 插入后返回主键

public Integer insertAndReturnId(String loginName, String password) {
  return useDsl()
      .insertInto(SYSTEM_ADMIN)
      .set(SYSTEM_ADMIN.LOGIN_NAME, loginName)
      .set(SYSTEM_ADMIN.PASSWORD, password)
      .returning(SYSTEM_ADMIN.ID)
      .fetchOne(SYSTEM_ADMIN.ID);
}

说明

这里不是先插入、再额外查询,而是借助 PostgreSQL 的 returning 一次完成。

这种写法比单纯依赖 JDBC 自动回填主键更显式。


5.2 插入后返回整条 Record

public SystemAdminRecord insertAndReturnRecord(String loginName, String password) {
  return useDsl()
      .insertInto(SYSTEM_ADMIN)
      .set(SYSTEM_ADMIN.LOGIN_NAME, loginName)
      .set(SYSTEM_ADMIN.PASSWORD, password)
      .returning()
      .fetchOne();
}

适用场景

  • 想拿到完整插入结果
  • 数据库有默认值、触发器、更新时间字段等
  • 想看到最终落库后的真实记录

5.3 插入后返回 POJO

public demo.jooq.gen.tables.pojos.SystemAdmin insertAndReturnPojo(String loginName, String password) {
  return useDsl()
      .insertInto(SYSTEM_ADMIN)
      .set(SYSTEM_ADMIN.LOGIN_NAME, loginName)
      .set(SYSTEM_ADMIN.PASSWORD, password)
      .returning()
      .fetchOneInto(demo.jooq.gen.tables.pojos.SystemAdmin.class);
}

这种写法很适合 DAO 对外返回纯数据对象。


5.4 UPSERT 后返回结果

PostgreSQL 的 on conflict ... do update 同样可以搭配 returning。

public SystemAdminRecord upsertAndReturnRecord(String loginName, String password) {
  return useDsl()
      .insertInto(SYSTEM_ADMIN)
      .set(SYSTEM_ADMIN.LOGIN_NAME, loginName)
      .set(SYSTEM_ADMIN.PASSWORD, password)
      .onConflict(SYSTEM_ADMIN.LOGIN_NAME)
      .doUpdate()
      .set(SYSTEM_ADMIN.PASSWORD, DSL.excluded(SYSTEM_ADMIN.PASSWORD))
      .returning()
      .fetchOne();
}

这非常适合“写入后立刻拿最终状态”的业务。


六、returning 和 Record.insert() 回填主键的区别

前一篇里我们讲过:

record.insert();
record.getId();

很多情况下也能拿到主键。

但两者还是有区别的。


1. record.insert() 后取主键

优点:

  • 写法自然
  • 对单行 Record 插入很方便

局限:

  • 更依赖底层驱动 / 方言 / jOOQ 配置的默认处理
  • 返回范围通常主要是回填 identity 字段

2. returning(...)

优点:

  • 语义更明确
  • 可返回任意字段
  • 可返回整条记录
  • 很适合 PostgreSQL 场景

所以在 PostgreSQL 中,如果你明确需要返回值,通常更推荐:

优先使用 returning(...)。


七、批量更新怎么做

批量更新的“批量”,其实有几种不同含义,必须区分清楚。


1. 相同 SQL 模板,多组参数

例如批量把多个人的密码更新成不同值:

  • 每条记录条件不同
  • 每条记录更新值也不同

这种适合 JDBC batch 风格。


2. 一条 SQL,更新多行

例如:

  • 把某一组用户状态统一改成禁用
  • 把一批 id 对应的数据统一标记删除

这种适合单条 update ... where in (...)


3. 更复杂的批量差异更新

例如:

  • 每行更新不同列
  • 需要 case when
  • 或借助临时表 / CTE

这就属于高级批量更新场景。


八、同条件批量更新

先看最简单、也最常见的一种。

8.1 一组 id 统一更新

public int batchUpdatePassword(List<Integer> ids, String newPassword) {
  if (ids == null || ids.isEmpty()) {
    return 0;
  }

  return useDsl()
      .update(SYSTEM_ADMIN)
      .set(SYSTEM_ADMIN.PASSWORD, newPassword)
      .where(SYSTEM_ADMIN.ID.in(ids))
      .execute();
}

特点

  • 一条 SQL
  • 性能通常很好
  • 适合统一改同一个值

这是最推荐优先考虑的批量更新方式。

因为它最简单,也最容易让数据库优化。


九、差异化批量更新:JDBC Batch 风格

如果每条记录要更新成不同值,例如:

  • id=1 改成 a
  • id=2 改成 b
  • id=3 改成 c

这时通常可以用 jOOQ 的 batch。


9.1 批量提交多条 update

public int[] batchUpdateDifferentPassword(List<demo.jooq.gen.tables.pojos.SystemAdmin> list) {
  if (list == null || list.isEmpty()) {
    return new int[0];
  }

  List<org.jooq.Query> queries = new ArrayList<>();

  for (demo.jooq.gen.tables.pojos.SystemAdmin pojo : list) {
    org.jooq.Query query = useDsl()
        .update(SYSTEM_ADMIN)
        .set(SYSTEM_ADMIN.PASSWORD, pojo.getPassword())
        .where(SYSTEM_ADMIN.ID.eq(pojo.getId()));
    queries.add(query);
  }

  return useDsl().batch(queries).execute();
}

说明

这里的语义是:

  • 每条更新 SQL 都独立
  • 但通过 JDBC batch 一次批处理提交

适合:

  • 每行更新值不同
  • 逻辑仍然相对简单

9.2 基于 Record 的批量存储

如果你拿到的是 Record 集合,也可以考虑:

public int[] batchStoreRecords(List<SystemAdminRecord> records) {
  if (records == null || records.isEmpty()) {
    return new int[0];
  }

  return useDsl().batchStore(records).execute();
}

注意

batchStore 的语义依赖每个 Record 的状态:

  • 有些会 insert
  • 有些会 update

所以它方便,但不一定是最清晰的方式。

团队协作时,如果希望 SQL 意图更明确,通常还是:

  • 批量 insert 用 batchInsert
  • 批量 update 用 batch(query...)

更容易读懂。


十、复杂差异更新:CASE WHEN

如果想把多行更新压缩成一条 SQL,可以考虑 case when。

例如按 id 更新不同密码:

public int updatePasswordByCase(List<demo.jooq.gen.tables.pojos.SystemAdmin> list) {
  if (list == null || list.isEmpty()) {
    return 0;
  }

  org.jooq.CaseConditionStep<String> caseStep = null;
  List<Integer> ids = new ArrayList<>();

  for (demo.jooq.gen.tables.pojos.SystemAdmin pojo : list) {
    ids.add(pojo.getId());
    if (caseStep == null) {
      caseStep = DSL.when(SYSTEM_ADMIN.ID.eq(pojo.getId()), pojo.getPassword());
    } else {
      caseStep = caseStep.when(SYSTEM_ADMIN.ID.eq(pojo.getId()), pojo.getPassword());
    }
  }

  return useDsl()
      .update(SYSTEM_ADMIN)
      .set(SYSTEM_ADMIN.PASSWORD, caseStep.otherwise(SYSTEM_ADMIN.PASSWORD))
      .where(SYSTEM_ADMIN.ID.in(ids))
      .execute();
}

特点

优点:

  • 一条 SQL 完成
  • 减少数据库往返

缺点:

  • SQL 会变长
  • 可读性会下降
  • 不适合非常大的批次

通常只有在明确有性能需求时才建议这样做。


十一、批量插入再深入一点

前一篇已经讲了 batchInsert,这里再补充两类实际工程建议。


11.1 用 batchInsert 批量导入 POJO

public int[] batchInsert(List<demo.jooq.gen.tables.pojos.SystemAdmin> pojos) {
  if (pojos == null || pojos.isEmpty()) {
    return new int[0];
  }

  List<SystemAdminRecord> records = new ArrayList<>();

  for (demo.jooq.gen.tables.pojos.SystemAdmin pojo : pojos) {
    SystemAdminRecord record = useDsl().newRecord(SYSTEM_ADMIN, pojo);
    records.add(record);
  }

  return useDsl().batchInsert(records).execute();
}

11.2 分批导入

public void batchInsertByChunk(List<demo.jooq.gen.tables.pojos.SystemAdmin> pojos, int chunkSize) {
  if (pojos == null || pojos.isEmpty()) {
    return;
  }

  int size = pojos.size();
  for (int i = 0; i < size; i += chunkSize) {
    int end = Math.min(i + chunkSize, size);
    List<demo.jooq.gen.tables.pojos.SystemAdmin> sub = pojos.subList(i, end);
    batchInsert(sub);
  }
}

为什么推荐分批

原因很简单:

  • 避免单批过大占用内存
  • 避免 JDBC batch 过长
  • 避免数据库压力瞬时过高
  • 出错时更容易定位

一般实践里,常见起步值是:

  • 200
  • 500
  • 1000

具体还是要压测。


十二、插入多行并返回结果

这里要注意一个现实问题:

批量 insert 后“逐行返回所有主键”这件事,并不是所有数据库 / 驱动 / 批处理方式都天然适合。

所以在 PostgreSQL 中,如果业务强依赖“插入后必须立刻拿到所有返回值”,通常有两种思路:


1. 小批量逐条 insert + returning

优点:

  • 返回值最明确
  • 编码简单

缺点:

  • 往返次数更多

2. 构造多 values 插入 + returning

jOOQ 可以构造单条多 values 的 insert,再 returning()。

示意写法如下:

public List<SystemAdminRecord> insertManyAndReturn(List<demo.jooq.gen.tables.pojos.SystemAdmin> pojos) {
  if (pojos == null || pojos.isEmpty()) {
    return List.of();
  }

  var step = useDsl()
      .insertInto(SYSTEM_ADMIN, SYSTEM_ADMIN.LOGIN_NAME, SYSTEM_ADMIN.PASSWORD);

  var valuesStep = step.values(pojos.get(0).getLoginName(), pojos.get(0).getPassword());

  for (int i = 1; i < pojos.size(); i++) {
    valuesStep = valuesStep.values(pojos.get(i).getLoginName(), pojos.get(i).getPassword());
  }

  return valuesStep
      .returning()
      .fetch();
}

适用场景

  • 一次插入数量不大
  • 明确需要返回每行结果
  • PostgreSQL 场景

不过这类写法在数据量很大时就不一定适合了。


十三、带返回值的 update / delete

PostgreSQL 不只是 insert 支持 returning,update / delete 也支持。

这在很多业务场景里非常有用。


13.1 update 后返回更新结果

public SystemAdminRecord updatePasswordAndReturn(Integer id, String newPassword) {
  return useDsl()
      .update(SYSTEM_ADMIN)
      .set(SYSTEM_ADMIN.PASSWORD, newPassword)
      .where(SYSTEM_ADMIN.ID.eq(id))
      .returning()
      .fetchOne();
}

适合场景

  • 更新后要拿最新记录
  • 省去再查一次
  • 想确认最终值

13.2 delete 后返回被删除记录

public SystemAdminRecord deleteAndReturn(Integer id) {
  return useDsl()
      .deleteFrom(SYSTEM_ADMIN)
      .where(SYSTEM_ADMIN.ID.eq(id))
      .returning()
      .fetchOne();
}

这在做审计日志、回收站、删除前备份时非常方便。


十四、高级 SQL:子查询更新

复杂业务里,经常会有“根据子查询结果来更新”的需求。

jOOQ 在这方面非常自然。

例如:

public int updatePasswordByLoginNameSubquery(String loginName, String newPassword) {
  return useDsl()
      .update(SYSTEM_ADMIN)
      .set(SYSTEM_ADMIN.PASSWORD, newPassword)
      .where(SYSTEM_ADMIN.ID.in(
          useDsl()
              .select(SYSTEM_ADMIN.ID)
              .from(SYSTEM_ADMIN)
              .where(SYSTEM_ADMIN.LOGIN_NAME.eq(loginName))
      ))
      .execute();
}

虽然这个例子比较简单,但它说明了一个要点:

jOOQ 的子查询和主查询可以自然拼接,且全部保持强类型。


十五、高级 SQL:存在性判断

有些业务只关心“是否存在”,不需要真正查出整行。

这时不要先 fetchOne() 再判断非空,更推荐直接让数据库做存在性判断。

public boolean existsByLoginName(String loginName) {
  return useDsl()
      .fetchExists(
          useDsl()
              .selectOne()
              .from(SYSTEM_ADMIN)
              .where(SYSTEM_ADMIN.LOGIN_NAME.eq(loginName))
      );
}

优点

  • 语义更明确
  • 数据库优化更直接
  • 不需要真的把整行拉回应用层

十六、高级 SQL:聚合查询

jOOQ 不只是 CRUD,也很适合统计查询。

例如查询管理员总数:

public int countAll() {
  return useDsl()
      .selectCount()
      .from(SYSTEM_ADMIN)
      .fetchOne(0, int.class);
}

例如按密码分组统计数量:

public List<org.jooq.Record2<String, Integer>> countGroupByPassword() {
  return useDsl()
      .select(
          SYSTEM_ADMIN.PASSWORD,
          DSL.count().cast(Integer.class)
      )
      .from(SYSTEM_ADMIN)
      .groupBy(SYSTEM_ADMIN.PASSWORD)
      .fetch();
}

如果不想直接返回 Record2,也可以映射到自定义 DTO。


十七、高级 SQL:表达式字段与别名

有时需要查询一个计算字段,例如拼接文本:

public List<org.jooq.Record2<Integer, String>> selectDisplayName() {
  return useDsl()
      .select(
          SYSTEM_ADMIN.ID,
          DSL.concat(DSL.inline("admin:"), SYSTEM_ADMIN.LOGIN_NAME).as("display_name")
      )
      .from(SYSTEM_ADMIN)
      .fetch();
}

这说明 jOOQ 并不是只能查“表原字段”,也非常适合构建表达式列。


十八、推荐的 DAO 示例

下面给出一个把本文重点串起来的 DAO 示例。

package demo.jooq.dao;

import static demo.jooq.gen.tables.SystemAdmin.SYSTEM_ADMIN;

import java.util.ArrayList;
import java.util.List;

import org.jooq.DSLContext;
import org.jooq.Query;
import org.jooq.impl.DSL;

import com.litongjava.annotation.Inject;

import demo.jooq.gen.tables.pojos.SystemAdmin;
import demo.jooq.gen.tables.records.SystemAdminRecord;
import demo.jooq.tx.TransactionContext;

public class SystemAdminAdvancedDao {

  @Inject
  private DSLContext dsl;

  private DSLContext useDsl() {
    DSLContext txDsl = TransactionContext.get();
    return txDsl != null ? txDsl : dsl;
  }

  public Integer insertAndReturnId(String loginName, String password) {
    return useDsl()
        .insertInto(SYSTEM_ADMIN)
        .set(SYSTEM_ADMIN.LOGIN_NAME, loginName)
        .set(SYSTEM_ADMIN.PASSWORD, password)
        .returning(SYSTEM_ADMIN.ID)
        .fetchOne(SYSTEM_ADMIN.ID);
  }

  public SystemAdminRecord insertAndReturnRecord(String loginName, String password) {
    return useDsl()
        .insertInto(SYSTEM_ADMIN)
        .set(SYSTEM_ADMIN.LOGIN_NAME, loginName)
        .set(SYSTEM_ADMIN.PASSWORD, password)
        .returning()
        .fetchOne();
  }

  public SystemAdminRecord upsertAndReturn(String loginName, String password) {
    return useDsl()
        .insertInto(SYSTEM_ADMIN)
        .set(SYSTEM_ADMIN.LOGIN_NAME, loginName)
        .set(SYSTEM_ADMIN.PASSWORD, password)
        .onConflict(SYSTEM_ADMIN.LOGIN_NAME)
        .doUpdate()
        .set(SYSTEM_ADMIN.PASSWORD, DSL.excluded(SYSTEM_ADMIN.PASSWORD))
        .returning()
        .fetchOne();
  }

  public int batchUpdatePassword(List<Integer> ids, String password) {
    if (ids == null || ids.isEmpty()) {
      return 0;
    }

    return useDsl()
        .update(SYSTEM_ADMIN)
        .set(SYSTEM_ADMIN.PASSWORD, password)
        .where(SYSTEM_ADMIN.ID.in(ids))
        .execute();
  }

  public int[] batchUpdateDifferentPassword(List<SystemAdmin> list) {
    if (list == null || list.isEmpty()) {
      return new int[0];
    }

    List<Query> queries = new ArrayList<>();
    for (SystemAdmin pojo : list) {
      Query query = useDsl()
          .update(SYSTEM_ADMIN)
          .set(SYSTEM_ADMIN.PASSWORD, pojo.getPassword())
          .where(SYSTEM_ADMIN.ID.eq(pojo.getId()));
      queries.add(query);
    }

    return useDsl().batch(queries).execute();
  }

  public int[] batchInsert(List<SystemAdmin> pojos) {
    if (pojos == null || pojos.isEmpty()) {
      return new int[0];
    }

    List<SystemAdminRecord> records = new ArrayList<>();
    for (SystemAdmin pojo : pojos) {
      SystemAdminRecord record = useDsl().newRecord(SYSTEM_ADMIN, pojo);
      records.add(record);
    }

    return useDsl().batchInsert(records).execute();
  }

  public boolean existsByLoginName(String loginName) {
    return useDsl().fetchExists(
        useDsl()
            .selectOne()
            .from(SYSTEM_ADMIN)
            .where(SYSTEM_ADMIN.LOGIN_NAME.eq(loginName))
    );
  }

  public SystemAdminRecord updateAndReturn(Integer id, String password) {
    return useDsl()
        .update(SYSTEM_ADMIN)
        .set(SYSTEM_ADMIN.PASSWORD, password)
        .where(SYSTEM_ADMIN.ID.eq(id))
        .returning()
        .fetchOne();
  }

  public SystemAdminRecord deleteAndReturn(Integer id) {
    return useDsl()
        .deleteFrom(SYSTEM_ADMIN)
        .where(SYSTEM_ADMIN.ID.eq(id))
        .returning()
        .fetchOne();
  }
}

十九、UPSERT 与事务边界的关系

虽然 PostgreSQL UPSERT 本身是一条原子 SQL,但它不意味着整个业务方法就不需要事务。

例如下面这种业务:

  • UPSERT 管理员
  • 再插入审计日志
  • 再更新别的表

如果这三步要求“一起成功或一起失败”,那仍然应该放在 service 层事务里。

例如:

@ATransactional
public void createOrUpdateAdmin(String loginName, String password) {
  systemAdminAdvancedDao.upsertAndReturn(loginName, password);
  auditLogDao.insert("system_admin upsert");
}

要记住:

单条 SQL 的原子性,不等于整个业务流程的事务一致性。


二十、什么时候优先用 UPSERT

推荐优先考虑 UPSERT 的场景:

1. 存在明确唯一键

例如:

  • login_name
  • email
  • biz_code

2. 业务允许“有则改,无则增”

例如:

  • 同步外部系统主数据
  • 账号初始化
  • 配置项覆盖写入

3. 有并发写入可能

UPSERT 比“先查再写”更稳。


二十一、什么时候不建议滥用 UPSERT

以下场景要谨慎:

1. 业务规则复杂

例如:

  • 已存在时不是简单更新
  • 需要比较旧值
  • 需要检查状态流转是否合法

这时通常应该:

  • 先查
  • 再做业务判断
  • 再决定更新方式

2. 需要非常明确地区分“新增成功”和“更新成功”

UPSERT 虽然方便,但会把两种语义融合在一起。

如果业务上两者是不同事件,那最好不要直接混写。


二十二、几个常见坑


1. UPSERT 必须依赖唯一键或主键

没有冲突约束,onConflict(...) 就没有意义。

所以用之前一定要确认数据库表结构已经具备:

  • 主键
  • 唯一约束
  • 或唯一索引

2. 批量更新不一定要用 batch

如果是“一批 id 统一改同一个值”,优先用:

update ... where id in (...)

这通常比 batch 多条 update 更自然。


3. returning() 很强,但不要无脑全返回

如果你只需要主键,就写:

returning(SYSTEM_ADMIN.ID)

而不是默认整行返回。

这样更清晰,也更节省数据传输。


4. batchStore() 虽方便,但意图不如显式 batch 清楚

批量 insert、批量 update、混合 save,最好分开表达。

可读性会更好。


5. PostgreSQL 特性很强,但要承认它是方言能力

像:

  • returning
  • on conflict

都带有 PostgreSQL 方言色彩。

这不是缺点,但要意识到:

如果项目目标是深度利用 PostgreSQL,那就应该主动拥抱方言能力,而不是假装自己在写“数据库无关 SQL”。

这恰好也是 jOOQ 非常适合的地方。


二十三、总结

这一篇的核心,其实可以压缩成一句话:

在 tio-boot + jOOQ + PostgreSQL 体系中,不要只把 jOOQ 当成 CRUD 工具,而要把它当成“强类型的 PostgreSQL SQL 表达层”。

通过本文,我们完成了:

  • 使用 onConflict(...).doUpdate() 实现 PostgreSQL UPSERT
  • 使用 returning(...) 获取主键、Record、POJO
  • 区分应用层 saveOrUpdate 和数据库层 UPSERT
  • 掌握同值批量更新与差异化批量更新
  • 理解 batchInsert、batch(query...)、batchStore 的差异
  • 使用 fetchExists、聚合查询、表达式字段扩展高级 SQL 能力

到这里,整个 tio-boot + jOOQ 体系已经具备了相当完整的数据库访问能力:

  • 强类型表字段引用
  • Record / POJO 分层
  • Service 层事务边界
  • PostgreSQL 原生 UPSERT
  • 批量写入与更新
  • 返回主键与返回整行
  • 高级 SQL 表达能力

一句话总结:

jOOQ 的真正价值,不只是替代 XML,而是让你在 Java 中优雅地保留数据库原生能力。

Edit this page
Last Updated: 3/14/26, 2:58 PM
Contributors: litongjava
Prev
基于 Record / POJO 增删改查
Next
的多表关联查询、DTO 投影、聚合统计与视图封装