接口幂等性详解
1. 什么是幂等性
幂等性指的是对同一个操作的多次执行所产生的影响与一次执行的影响相同。无论操作执行多少次,系统状态都应该保持一致。
在计算机科学和网络领域中,幂等性通常用来描述服务或操作的特性。对于RESTful API或HTTP方法,一个幂等操作的特点是:对于相同的输入,在多次执行之后,得到的结果是一致的。
例如,HTTP中的GET请求是幂等的,无论请求多少次,结果都是相同的。而POST请求则不一定是幂等的,因为每次执行时,可能会创建新的资源或状态,从而改变系统状态。
在处理分布式系统和网络通信时,保证操作的幂等性是非常重要的。因为网络中的请求可能因为重试、超时、丢包等原因被多次发送,而服务端必须能够正确处理这些重复的请求,而不会因为重复执行导致系统状态异常或资源多次创建等问题。
确保API的幂等性是设计和开发高质量、健壮系统的一个关键因素,可以减少因为重复操作导致的副作用,保障系统的可靠性和稳定性。
2. 什么是接口幂等性
接口幂等性是指针对同一个操作,无论执行多少次,系统的状态都保持一致。在Web开发中,特别是对于RESTful API,接口的幂等性是一种非常重要的性质。
一个幂等操作满足以下条件:
- 多次执行相同操作,产生的效果与执行一次相同操作的效果相同。
- 重复执行相同请求的结果与第一次执行该请求的结果相同。
这意味着无论对一个接口发送多少次相同的请求,系统都保持一致的状态,不会因为重复请求而导致状态的变化或资源的重复创建。在幂等性的设计中,服务端需要保证接收到重复请求时,不会因为重复执行操作而引起问题。
幂等性的设计对于解决由于网络延迟、超时、重试、系统问题等导致重复请求的情况非常重要。例如,对于一次性支付的操作,幂等性保证了即使用户在网络不稳定时多次点击支付按钮,系统也只会执行一次支付操作,避免重复扣款。
在RESTful API中,GET请求通常是幂等的,因为它只是获取资源而不会对资源状态做出改变;而POST请求一般不是幂等的,因为每次执行可能会创建新资源或改变系统状态。
确保接口的幂等性对于系统的正确性、可靠性和稳定性至关重要。

3. 如果忽略接口幂等性设计会出现哪些问题
忽略接口的幂等性设计可能导致以下问题:
-
重复操作导致资源状态异常: 如果接口不具有幂等性,重复执行同一个请求可能会导致资源状态的不一致或异常,例如多次创建相同的订单、重复扣款等。
-
重复创建资源: 对于非幂等的操作,可能会导致多次执行请求创建重复的资源,例如多次支付,每次请求都会创建新的支付记录。
-
系统副作用: 非幂等操作可能会对系统状态造成非预期的副作用,例如多次发送邮件、多次触发同一个事件,导致系统状态不一致。
-
用户体验下降: 如果接口不是幂等的,用户在网络波动或系统问题下多次发送相同请求时,可能会产生用户体验问题,例如重复下单、重复支付等。
-
系统负载增加: 重复请求可能导致系统负载增加,对服务器和数据库等资源产生不必要的压力,可能导致系统性能下降或宕机。
综上所述,忽略接口的幂等性设计可能导致系统状态不一致、资源重复创建、用户体验问题和系统负载增加等诸多问题,对系统的正确性和稳定性造成威胁。因此,确保接口的幂等性是设计和开发过程中的重要考虑因素。
4. 如何设计接口幂等性
4.1 使用Token+Redis
使用Token和Redis实现接口的幂等性通常涉及以下步骤:
- 生成并传递 Token:
前端生成 Token: 在发送请求之前,前端生成一个唯一的 Token,通常是一个随机生成的字符串,比如 UUID,并将这个Token以客户ID为主键,以Token为值放入Redis
Token 传递: 将 Token 放置于请求的 Header 或 Body 中。 - 服务端处理 Token:
服务端接收 Token: 接收到请求后,服务端从请求中提取 Token。 - 验证 Token:
检查 Token 是否存在于 Redis 中: 使用 Redis 数据库验证该 Token 是否已存在于数据库中。 - 保证幂等性:
Token 存在: 如果 Token 已存在于 Redis 中,表示该请求已被处理过,服务端直接返回已存储的结果,不执行重复操作。(或者处理业务,并删除Token)
Token 不存在: 如果 Token 不存在于 Redis 中,表示这是一个新的请求,服务端执行相应的业务逻辑,并将 Token 存储到 Redis,以防止重复请求。(或者直接返回)
示例伪代码:
以下是一个简单的伪代码示例,演示了如何使用 Token+Redis 来确保接口的幂等性:
import redis.clients.jedis.Jedis;public class IdempotentAPIHandler {private Jedis jedis;public IdempotentAPIHandler() {this.jedis = new Jedis("localhost", 6379); // 连接 Redis}public String processRequest(String token, String requestData) {if (checkTokenInRedis(token)) {// Token 存在,直接返回已存储的处理结果return jedis.get(token);} else {// Token 不存在,处理请求并将结果存储到 RedisString result = processRequestData(requestData); // 处理请求数据,获取结果jedis.set(token, result, "NX", "EX", 3600); // 设置 Token 并存储结果,设置过期时间为 1 小时return result;}}public boolean checkTokenInRedis(String token) {return jedis.exists(token); // 检查 Token 是否存在于 Redis}public String processRequestData(String requestData) {// 处理请求数据,返回处理结果return "Processed data: " + requestData;}public String generateToken() {// 生成 Token,通常是一个唯一的字符串return "generated_token"; // 这里可以使用 UUID.randomUUID().toString() 等生成唯一标识}public static void main(String[] args) {IdempotentAPIHandler handler = new IdempotentAPIHandler();String token = handler.generateToken(); // 生成 Token// 模拟处理 API 请求String requestData = "some_data";String result = handler.processRequest(token, requestData);System.out.println("Result: " + result);}
}
以上伪代码演示了如何在Java中使用 Token+Redis 实现接口的幂等性。在实际应用中,你需要根据具体的业务需求和框架,对这个示例进行适当调整。
4.2 使用数据库唯一索引
使用数据库唯一索引来实现接口的幂等性是一种可行的方法。这通常涉及在数据库表中创建唯一索引,以确保请求的唯一性。以下是一些步骤和概念:
-
设计数据库表结构:
在数据库中创建一张表来记录请求,可以包括请求的唯一标识、请求内容以及处理结果等字段。 -
创建唯一索引:
使用数据库的唯一索引特性,将请求的唯一标识字段设置为唯一索引。这样,当重复插入具有相同唯一标识的请求时,数据库会拒绝插入重复的记录。 -
插入请求前进行唯一索引检查:
在处理请求前,首先查询数据库检查要插入的唯一标识是否已存在。如果存在,则表明这个请求已经被处理过,可以直接返回已存储的结果。 -
处理请求并插入数据库:
如果唯一索引检查结果表明这个请求是新的,可以执行相应的业务逻辑,并将请求的唯一标识及处理结果插入到数据库中。
示例伪代码:
以下是一个简单的伪代码示例,演示了如何使用数据库的唯一索引来确保接口的幂等性:
// 假设这里是数据库操作对象,这里使用伪代码表示
class DatabaseOperation {// 执行插入操作,以请求的唯一标识作为唯一索引public boolean insertRequest(String uniqueIdentifier, String requestData) {// 在数据库中执行插入操作,使用唯一索引来保证唯一性// 如果遇到唯一索引重复错误(比如DuplicateEntry),说明请求已存在,返回 false// 如果成功插入,说明请求是新的,返回 true}// 根据唯一标识查询请求public String getRequestResult(String uniqueIdentifier) {// 根据唯一标识从数据库中获取请求处理结果}
}// 在处理请求的服务类中
class IdempotentAPIHandler {private DatabaseOperation db;public IdempotentAPIHandler() {this.db = new DatabaseOperation(); // 初始化数据库操作对象}public String processRequest(String uniqueIdentifier, String requestData) {// 检查请求是否已存在if (db.getRequestResult(uniqueIdentifier) != null) {// 请求已存在,直接返回结果return db.getRequestResult(uniqueIdentifier);} else {// 请求是新的,处理请求并将结果插入到数据库String result = processRequestData(requestData); // 处理请求数据,获取结果db.insertRequest(uniqueIdentifier, result); // 插入请求到数据库return result;}}// 其他方法和业务逻辑
}
这个示例演示了如何利用数据库的唯一索引来确保接口的幂等性。在实际应用中,你需要根据具体的数据库和业务需求,调整这个示例。
4.3 使用分布式锁
使用分布式锁来确保接口的幂等性通常会涉及下列步骤:
-
获取分布式锁:
在处理请求之前,尝试获取一个分布式锁,以确保在处理请求期间不会有其他请求来干扰。这样可以确保仅有一个请求能够进入关键的临界区。 -
处理请求:
如果获取了分布式锁,就可以执行请求处理逻辑。这可以是具体的业务逻辑,确保在这个锁的作用下完成了唯一的处理操作。 -
释放分布式锁:
在处理结束后,释放分布式锁,以允许其他请求进入临界区,继续执行接下来的操作。
示例伪代码:
以下是一个使用分布式锁实现接口幂等性的简单示例:
import redis.clients.jedis.Jedis;public class IdempotentAPIHandler {private Jedis jedis;private String lockKey = "request_lock";public IdempotentAPIHandler() {this.jedis = new Jedis("localhost", 6379); // 连接 Redis}public String processRequest(String requestData) {String lockValue = acquireDistributedLock(); // 尝试获取分布式锁if (lockValue != null) {try {// 成功获取锁,处理请求String result = processRequestData(requestData); // 处理请求数据,获取结果return result;} finally {releaseDistributedLock(lockValue); // 处理结束后释放锁}} else {// 获取锁失败,可能有其他请求正在处理中return "Request is being processed. Please try again later.";}}public String acquireDistributedLock() {// 尝试获取分布式锁String lockValue = UUID.randomUUID().toString(); // 生成唯一的锁值String result = jedis.set(lockKey, lockValue, "NX", "EX", 60); // 设置锁,并设置过期时间if ("OK".equals(result)) {return lockValue; // 获取锁成功} else {return null; // 获取锁失败}}public void releaseDistributedLock(String lockValue) {// 释放分布式锁String currentValue = jedis.get(lockKey);if (currentValue != null && currentValue.equals(lockValue)) {jedis.del(lockKey); // 删除锁}}public String processRequestData(String requestData) {// 处理请求数据获取结果return "Processed data: " + requestData;}// 其他方法和业务逻辑
}
这个示例演示了如何使用Redis的分布式锁来确保在处理请求时只有一个请求能够进入关键的临界区,从而保证接口的幂等性。在实际应用中,你需要根据具体的业务需求和框架,对这个示例进行适当调整。
5. 案例:使用Token+Redis保证提交订单接口的幂等性
当设计一个电商网站的提交订单接口时,确保该接口的幂等性对于避免重复订单、重复支付等问题非常重要。以下是一个示例,使用Token+Redis实现提交订单接口的幂等性:
import redis.clients.jedis.Jedis;
import java.util.UUID;public class OrderService {private Jedis jedis;public OrderService() {this.jedis = new Jedis("localhost", 6379); // 连接 Redis}public String submitOrder(String userId, String product) {String token = generateToken(userId, product); // 生成订单的唯一 Tokenif (checkIfOrderExists(token)) {// 订单已存在,直接返回已处理的结果return "Order already submitted.";} else {// 订单不存在,处理订单并存储到 RedisString result = processOrder(userId, product); // 处理订单逻辑,获取结果// 将订单信息存储到 Redis,并设置过期时间为一段合理时间,比如 24 小时jedis.set(token, result, "NX", "EX", 86400);return "Order submitted successfully.";}}private String generateToken(String userId, String product) {// 生成 Token,这里用 userId 和商品信息作为组合生成的 Tokenreturn "order_" + userId + "_" + product + "_" + UUID.randomUUID().toString();}private boolean checkIfOrderExists(String token) {// 检查订单是否已存在于 Redis 中return jedis.exists(token);}private String processOrder(String userId, String product) {// 处理订单的逻辑,可以包括数据库插入、支付等步骤return "Processed Order: " + product + " for User: " + userId;}public static void main(String[] args) {OrderService orderService = new OrderService();// 模拟订单提交String userId = "user123";String product = "ProductABC";String submissionResult = orderService.submitOrder(userId, product);System.out.println(submissionResult);}
}
这个示例演示了一个简单的订单提交接口,确保了订单的幂等性。在实际应用中,你需要根据实际业务需求和框架,对这个示例进行适当调整。
相关文章:
接口幂等性详解
1. 什么是幂等性 幂等性指的是对同一个操作的多次执行所产生的影响与一次执行的影响相同。无论操作执行多少次,系统状态都应该保持一致。 在计算机科学和网络领域中,幂等性通常用来描述服务或操作的特性。对于RESTful API或HTTP方法,一个幂…...
Java操作redis常见类型数据存储
一,Java连接Redis 1.1 导入依赖 打开IDEA在pom.xml导入依赖 注意:要在dependencies标签中导入 <dependency><groupId>redis.clients</groupId><artifactId>jedis</artifactId><version>2.9.0</version> &…...
【深度学习】pytorch——Autograd
笔记为自我总结整理的学习笔记,若有错误欢迎指出哟~ 深度学习专栏链接: http://t.csdnimg.cn/dscW7 pytorch——Autograd Autograd简介requires_grad计算图没有梯度追踪的张量ensor.data 、tensor.detach()非叶子节点的梯度计算图特点总结 利用Autograd实…...
【ARM 安全系列介绍 1 -- 奇偶校验与海明码校验详细介绍】
文章目录 奇偶校验介绍奇偶校验 python 实现奇偶校验C代码实现 海明码详细介绍 奇偶校验介绍 奇偶校验是一种错误检测方法,广泛应用于计算机内部以及数据通信领域。其基本原理是为了使得一组数据(通常是一字节8位)中的“1”的个数为偶数或奇…...
分享34个发布商会PPT,总有一款适合您
分享34个发布商会PPT,总有一款适合您 链接:https://pan.baidu.com/s/1jP9toqTZONWeDIcxvw1wxg?pwd8888 提取码:8888 Python采集代码下载链接:采集代码.zip - 蓝奏云 学习知识费力气,收集整理更不易。知识付费甚…...
047_第三代软件开发-日志分离
第三代软件开发-日志分离 文章目录 第三代软件开发-日志分离项目介绍日志分离用法 关键字: Qt、 Qml、 log、 日志、 分离 项目介绍 欢迎来到我们的 QML & C 项目!这个项目结合了 QML(Qt Meta-Object Language)和 C 的强…...
ChinaSoft 论坛巡礼 | 系统与网络安全论坛
2023年CCF中国软件大会(CCF ChinaSoft 2023)由CCF主办,CCF系统软件专委会、形式化方法专委会、软件工程专委会以及复旦大学联合承办,将于2023年12月1-3日在上海国际会议中心举行。 本次大会主题是“智能化软件创新推动数字经济与社…...
Ubuntu Gitlab安javascript:void(‘numberedlist‘)装
原因: 代码越改越多,越难维护,开发代码和发布代码融为一体;2人以上开发,都会修改代码,修改次数一多,代码难以维护 其中:前往Gitlab官网:gitlab/gitlab-ce - Packages pa…...
11.4-GPT4AllTools版本已开始对小部分GPT3.5用户内测推送
OpenAI已经开始小规模推送GPT4 AllTools功能,部分GPT博主已经第一时间体验了此功能,此功能特色是整合目前的多模态功能以及文件上传和联网模块,无需切换,更要全面综合 可上传包括 PDF、数据文件在内的任意文档,并进行分…...
竞赛选题 深度学习手势检测与识别算法 - opencv python
文章目录 0 前言1 实现效果2 技术原理2.1 手部检测2.1.1 基于肤色空间的手势检测方法2.1.2 基于运动的手势检测方法2.1.3 基于边缘的手势检测方法2.1.4 基于模板的手势检测方法2.1.5 基于机器学习的手势检测方法 3 手部识别3.1 SSD网络3.2 数据集3.3 最终改进的网络结构 4 最后…...
语言模型AI——聊聊GPT使用情形与影响
GPT的出现象征着人工智能自然语言处理技术的一次巨大飞跃。从编程助手到写作利器,它的身影在各个行业中越来越常见。百度【文心一言】、CSDN【C知道】等基于GPT的产品相继推出,让我们看到了其广泛的应用前景。然而,随着GPT的普及,…...
浅谈事件冒泡和事件捕获
事件冒泡和事件捕获分别由微软和网景公司提出,这两个概念都是为了解决页面中事件流(事件发生顺序)的问题。 <div id"div1"><div id"div2"><div id"div3">click</div></div> <…...
CSS 背景、文本、字体
CSS背景: CSS背景属性用于定义HTML元素的背景。CSS属性定义背景效果:background-color;background-image;background-repeat;background-attachment;background-position。 background-color属性定义元素…...
爬取Elastic Stack采集的Nginx内容
以下是一个简单的Go语言爬虫程序,用于爬取Elastic Stack采集的Nginx内容。请注意,这只是一个基本的示例,实际使用时可能需要根据具体情况进行修改和扩展。 package mainimport ("fmt""net/http""io/ioutil" )…...
python 机器学习 常用函数
一 np.random.randint "randint" 是 "random integer" 的缩写,表示生成随机整数。 np.random.randint 是 NumPy 库中的一个函数,用于生成随机整数。以下是该函数的一般语法: np.random.randint(low, high, size)其中…...
手写操作系统篇:环境配置
文章目录 前言C环境配置Rust环境配置Qemu安装 前言 这篇博客先配置好我们之后的开发环境,下载一些依赖的软件包 建议大家使用ubuntu操作系统 C环境配置 sudo apt-get update && sudo apt-get upgrade sudo apt-get install git wget build-essential gdb…...
地理信息系统原理-空间数据结构(7)
四叉树编码 1.四叉树编码定义 四叉树数据结构是一种对栅格数据的压缩编码方法,其基本思想是将一幅栅格数据层或图像等分为四部分,逐块检查其格网属性值(或灰度);如果某个子区的所有格网值都具有相同的值࿰…...
三国志14信息查询小程序(历史武将信息一览)制作更新过程03-主要页面的设计
1,小程序的默认显示 分为三部分,头部的标题、中间的内容区和底部的标签栏。点击标签可以切换不同页面,这是在app.json文件中配置的。代码如下: //所有用到的页面都需要在 pages 数组中列出,否则小程序可能会出现错误或…...
学习Opencv(蝴蝶书/C++)相关——2.用clang++或g++命令行编译程序
文章目录 1. c/cpp程序的执行1.1 cpp程序的编译过程1.2 预处理指令1.3 编译过程的细节2. macOS下使用Clang看cpp程序的编译过程2.1 示例2.1.1 第一步 预处理器-preprocessor2.1.2 第二步 编译器-compiler2.1.3 第三步 汇编器-assembler2.1.4 第四步 链接器-linker2.1.5 链接其他…...
【Unity细节】VS不能附加到Unity程序中解决方法大全
👨💻个人主页:元宇宙-秩沅 hallo 欢迎 点赞👍 收藏⭐ 留言📝 加关注✅! 本文由 秩沅 原创 😶🌫️收录于专栏:unity细节和bug 😶🌫️优质专栏 ⭐【…...
css实现圆环展示百分比,根据值动态展示所占比例
代码如下 <view class""><view class"circle-chart"><view v-if"!!num" class"pie-item" :style"{background: conic-gradient(var(--one-color) 0%,#E9E6F1 ${num}%),}"></view><view v-else …...
React Native 开发环境搭建(全平台详解)
React Native 开发环境搭建(全平台详解) 在开始使用 React Native 开发移动应用之前,正确设置开发环境是至关重要的一步。本文将为你提供一份全面的指南,涵盖 macOS 和 Windows 平台的配置步骤,如何在 Android 和 iOS…...
day52 ResNet18 CBAM
在深度学习的旅程中,我们不断探索如何提升模型的性能。今天,我将分享我在 ResNet18 模型中插入 CBAM(Convolutional Block Attention Module)模块,并采用分阶段微调策略的实践过程。通过这个过程,我不仅提升…...
UR 协作机器人「三剑客」:精密轻量担当(UR7e)、全能协作主力(UR12e)、重型任务专家(UR15)
UR协作机器人正以其卓越性能在现代制造业自动化中扮演重要角色。UR7e、UR12e和UR15通过创新技术和精准设计满足了不同行业的多样化需求。其中,UR15以其速度、精度及人工智能准备能力成为自动化领域的重要突破。UR7e和UR12e则在负载规格和市场定位上不断优化…...
实现弹窗随键盘上移居中
实现弹窗随键盘上移的核心思路 在Android中,可以通过监听键盘的显示和隐藏事件,动态调整弹窗的位置。关键点在于获取键盘高度,并计算剩余屏幕空间以重新定位弹窗。 // 在Activity或Fragment中设置键盘监听 val rootView findViewById<V…...
在WSL2的Ubuntu镜像中安装Docker
Docker官网链接: https://docs.docker.com/engine/install/ubuntu/ 1、运行以下命令卸载所有冲突的软件包: for pkg in docker.io docker-doc docker-compose docker-compose-v2 podman-docker containerd runc; do sudo apt-get remove $pkg; done2、设置Docker…...
「全栈技术解析」推客小程序系统开发:从架构设计到裂变增长的完整解决方案
在移动互联网营销竞争白热化的当下,推客小程序系统凭借其裂变传播、精准营销等特性,成为企业抢占市场的利器。本文将深度解析推客小程序系统开发的核心技术与实现路径,助力开发者打造具有市场竞争力的营销工具。 一、系统核心功能架构&…...
LangFlow技术架构分析
🔧 LangFlow 的可视化技术栈 前端节点编辑器 底层框架:基于 (一个现代化的 React 节点绘图库) 功能: 拖拽式构建 LangGraph 状态机 实时连线定义节点依赖关系 可视化调试循环和分支逻辑 与 LangGraph 的深…...
机器学习的数学基础:线性模型
线性模型 线性模型的基本形式为: f ( x ) ω T x b f\left(\boldsymbol{x}\right)\boldsymbol{\omega}^\text{T}\boldsymbol{x}b f(x)ωTxb 回归问题 利用最小二乘法,得到 ω \boldsymbol{\omega} ω和 b b b的参数估计$ \boldsymbol{\hat{\omega}}…...
Vue3 PC端 UI组件库我更推荐Naive UI
一、Vue3生态现状与UI库选择的重要性 随着Vue3的稳定发布和Composition API的广泛采用,前端开发者面临着UI组件库的重新选择。一个好的UI库不仅能提升开发效率,还能确保项目的长期可维护性。本文将对比三大主流Vue3 UI库(Naive UI、Element …...
