0

0

Python 幂等写入的多种实现方式

舞姬之光

舞姬之光

发布时间:2026-02-25 19:46:02

|

571人浏览过

|

来源于php中文网

原创

应优先使用 upsert 而非先查后写实现幂等写入,因其通过唯一约束在数据库层原子性处理“存在则更新、不存在则插入”,避免并发竞态;需配合唯一索引、精准捕获冲突错误码(如 postgresql 23505、mysql 1062),或结合 redis setnx + 请求 id 实现跨服务幂等。

python 幂等写入的多种实现方式

upsert 而不是先查后写

绝大多数“幂等写入”需求,本质是避免重复插入相同数据。直接查再判断再插入(SELECT + INSERT)在并发下必然出错——两个请求同时查不到,然后都插入成功。真正可靠的做法是把“存在则更新、不存在则插入”压进一条语句里。

PostgreSQL 用 ON CONFLICT,MySQL 用 INSERT ... ON DUPLICATE KEY UPDATE,SQLite 用 INSERT OR REPLACEINSERT OR IGNORE 配合后续 UPDATE。关键前提是:必须有唯一约束(UNIQUEPRIMARY KEY),否则 upsert 无从判断“重复”。

  • 别在应用层做 if not exists: insert() —— 这不是幂等,是竞态温床
  • 唯一约束字段要选业务上真正能标识“同一行”的列,比如 order_iduser_id+date 组合,而不是自增 id
  • MySQL 的 REPLACE INTO 会先删后插,触发 DELETEINSERT 双重触发器,且自增 ID 可能跳变,优先用 ON DUPLICATE KEY UPDATE

try/except 捕获唯一约束冲突最稳妥

ORM(如 SQLAlchemy、Django ORM)或原生驱动不总直接暴露 upsert 接口,尤其跨数据库时。此时更通用的做法是:直接 INSERT,靠数据库抛出的唯一约束错误来兜底。

PostgreSQL 错误码是 23505unique_violation),MySQL 是 1062ER_DUP_ENTRY)。捕获后不做任何写操作,即完成幂等。

立即学习Python免费学习笔记(深入)”;

BJXSHOP网上开店专家
BJXSHOP网上开店专家

BJXShop网上购物系统是一个高效、稳定、安全的电子商店销售平台,经过近三年市场的考验,在中国网购系统中属领先水平;完善的订单管理、销售统计系统;网站模版可DIY、亦可导入导出;会员、商品种类和价格均实现无限等级;管理员权限可细分;整合了多种在线支付接口;强有力搜索引擎支持... 程序更新:此版本是伴江行官方商业版程序,已经终止销售,现于免费给大家使用。比其以前的免费版功能增加了:1,整合了论坛

下载
  • 不要捕获宽泛的 Exception,只盯住明确的唯一冲突错误码,否则掩盖真实问题
  • Django 中可用 django.db.IntegrityError,但需进一步检查 args[0]__cause__.pgcode 才能区分是不是真因唯一键冲突
  • SQLAlchemy 用 exc.IntegrityError,配合 connection.execute(...).rowcount 判断是否真的没插入成功(有些驱动异常时 rowcount 仍为 1)

Redis SETNX + 数据库写入组合防重复提交

当“幂等”边界超出单条 SQL(比如一个下单请求含库存扣减 + 订单生成 + 消息投递),就得靠外部协调。Redis 的 SETNX(set if not exists)是最轻量的分布式锁前置手段。

流程是:用请求 ID(如 request_id 或签名摘要)作为 key,SETNX 成功才执行后续数据库操作;失败则直接返回已处理。注意过期时间必须设(EX 参数),否则锁残留会导致永久阻塞。

  • SETNX 后必须跟 EXPIRE,推荐合并用 SET key value EX 30 NX,避免中间状态丢失
  • key 命名要有业务上下文,比如 order:create:<code>sha256(user_id+goods_id+timestamp),别裸用 request_id(可能被重放)
  • 别在 Redis 成功后又去数据库查一遍“是否已存在”——这又回到竞态起点;SETNX 成功即代表首次进入,失败即代表已处理

HTTP 幂等性不能只靠后端,客户端也要配合

后端做再多,如果前端按钮重复点击、网络超时后盲目重试,照样产生多条记录。真正的幂等链路必须两端对齐。

关键动作是:客户端生成唯一请求标识(X-Idempotency-Key header),服务端用它做去重依据(存 Redis 或数据库临时表)。这个 key 必须由客户端生成并保证重试时复用,而不是服务端每次生成新 key。

  • 别让前端用时间戳或随机数当 Idempotency-Key——重试时变了就失效;应基于请求内容哈希,或由前端持久化一次后复用
  • 服务端收到重复 key 时,不能只返回 200,得返回原始响应体(比如第一次创建成功的订单 JSON),否则前端无法拿到数据
  • 该机制对 POST 有效,但对 GETPUTDELETE 本身具备幂等性,无需额外加 key

最难的从来不是写个 upsert 或捕获异常,而是厘清“哪一层该承担哪部分幂等责任”:数据库管单行唯一,Redis 管跨服务操作,HTTP 头管端到端请求生命周期。混用或漏掉任意一层,都会在某个并发密度下突然崩掉。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
数据分析工具有哪些
数据分析工具有哪些

数据分析工具有Excel、SQL、Python、R、Tableau、Power BI、SAS、SPSS和MATLAB等。详细介绍:1、Excel,具有强大的计算和数据处理功能;2、SQL,可以进行数据查询、过滤、排序、聚合等操作;3、Python,拥有丰富的数据分析库;4、R,拥有丰富的统计分析库和图形库;5、Tableau,提供了直观易用的用户界面等等。

1027

2023.10.12

SQL中distinct的用法
SQL中distinct的用法

SQL中distinct的语法是“SELECT DISTINCT column1, column2,...,FROM table_name;”。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

337

2023.10.27

SQL中months_between使用方法
SQL中months_between使用方法

在SQL中,MONTHS_BETWEEN 是一个常见的函数,用于计算两个日期之间的月份差。想了解更多SQL的相关内容,可以阅读本专题下面的文章。

379

2024.02.23

SQL出现5120错误解决方法
SQL出现5120错误解决方法

SQL Server错误5120是由于没有足够的权限来访问或操作指定的数据库或文件引起的。想了解更多sql错误的相关内容,可以阅读本专题下面的文章。

1842

2024.03.06

sql procedure语法错误解决方法
sql procedure语法错误解决方法

sql procedure语法错误解决办法:1、仔细检查错误消息;2、检查语法规则;3、检查括号和引号;4、检查变量和参数;5、检查关键字和函数;6、逐步调试;7、参考文档和示例。想了解更多语法错误的相关内容,可以阅读本专题下面的文章。

377

2024.03.06

oracle数据库运行sql方法
oracle数据库运行sql方法

运行sql步骤包括:打开sql plus工具并连接到数据库。在提示符下输入sql语句。按enter键运行该语句。查看结果,错误消息或退出sql plus。想了解更多oracle数据库的相关内容,可以阅读本专题下面的文章。

1415

2024.04.07

sql中where的含义
sql中where的含义

sql中where子句用于从表中过滤数据,它基于指定条件选择特定的行。想了解更多where的相关内容,可以阅读本专题下面的文章。

585

2024.04.29

sql中删除表的语句是什么
sql中删除表的语句是什么

sql中用于删除表的语句是drop table。语法为drop table table_name;该语句将永久删除指定表的表和数据。想了解更多sql的相关内容,可以阅读本专题下面的文章。

437

2024.04.29

batoto漫画官网入口与网页版访问指南
batoto漫画官网入口与网页版访问指南

本专题系统整理batoto漫画官方网站最新可用入口,涵盖最新官网地址、网页版登录页面及防走失访问方式说明,帮助用户快速找到batoto漫画官方平台,稳定在线阅读各类漫画内容。

127

2026.02.25

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 22.4万人学习

Django 教程
Django 教程

共28课时 | 4.5万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.7万人学习

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

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