微服务框架,Http异步编程中,如何保证数据的最终一致性
一、背景
在微服务框架下,跨服务之间的调用,当遇到操作耗时或者量大的情况,我们一般会采用异步编程实现。
本文出现的问题是:异步回调过来时,却未查询到数据库中的任务,导致未能正常处理回调。

下面是当时的日志信息:

根据此请求流水号查询数据库,得到入库时间:

总结下,产生这个错误的原因是数据库入库的时间晚于回调处理中查库的时间。
那么要如何避免这个问题呢?
有人可能要说,既然插入数据库的时间慢,是不是可以改为保存到redis呢?
要保证请求流水号存在,不一定要入库(redis也是为了实现插入快),所以我建议把它保存在jvm内存缓存中(具体实现可以是Map集合,也可以是caffeine)。
但是遇到生产环境分布式环境下,主服务是多节点部署的情况下,这个并不能解决问题。
所以还是调整代码的顺序,先入库,再发起http请求。(尽可能地做到先入库,实在不行,延迟1秒后再发起http请求)
当然,为了数据的实时性,不建议增加延时调用。因为这是偶现的极端情况。
二 、系统设计

如果外部服务从接收请求,到回调主服务,这个过程只有几十毫秒;也就是说,主服务收到外部服务的回调,是早于主服务把任务持久化到DB,在处理回调的时候,根据请求流水号无法查询到任务详情。
在极端的情况下,便出现了如下异常流程:

三、如何保证查询
要解决上面的异常流程,可以采用以下几种办法:
- 1、先保存任务表,再发起http请求。
- 2、在收到外部服务的回调后,第一时间未查询到,则查询第二次,第三次…… 尝试多次查询,等待保存任务表的插入语句操作完成。
- 3、主服务向外部服务发起主动查询操作结果。(外部服务提供一个根据请求流水号查询详情的接口)
1、调整发起请求的代码顺序
// 先保存至本地缓存
requestNoService.saveCache(requestNo, gson.toJson(request));// 保存至定时任务表
NotifyTasks notifyTasks = notifyTasksService.schedule(Constants.TaskCode.QUERY_CLASSROOM_COPY_RESULT, requestNo,assemblerCopyResultRequestUrl(requestNo), gson.toJson(request));// 再发起复制请求,避免出现回调收到了,但是本地表中没有数据,导致回调失败。
ApiResult<?> apiResult = notifyTasksService.notify(taskCode, notifyTasks.getRequestNo(),urlBuilder.toString(), gson.toJson(request));
修改前的代码:
// 发起复制请求
ApiResult<?> apiResult = notifyTasksService.notify(taskCode, requestNo, urlBuilder.toString(), gson.toJson(request));// 保存至定时任务表
notifyTasksService.schedule(Constants.TaskCode.CLASSROOM_COPY_RESULT, requestNo,assemblerCopyResultRequestUrl(requestNo), gson.toJson(request));
2、多次延迟查询
重试3次,每次延迟1秒后再发起查询
private String getNotifyParamsFromDB(String requestNo) {int attempt = 0;while (attempt < 3) {NotifyTasks notifyTasks = notifyTasksRepository.findTop1ByRequestNo(requestNo);if (null != notifyTasks) {return notifyTasks.getNotifyParams();}ThreadUtil.sleep(1, TimeUnit.SECONDS);attempt++;}return null;}
3、发起主动查询
定时任务,每隔5分钟,由主服务向外部服务发起主动查询。
这也是配套异步处理方案的常规处理动作。
比如交易系统里的支付操作,往往就需要商户主动向三方支付发起查询,询问支付中的订单状态其结果。
四、总结
异步编程,要求数据的一致性,往往采取的是最终一致性。
意味着在绝大多数情况下,实时回调可以保证数据更新的及时性,少数以及异常情况下,通过主动查询可以保证数据的最终一致性。
本案例中,主要是回调的速度快于本地数据库的入库,所以问题是少见的,一般也很难排查。
相关文章:
微服务框架,Http异步编程中,如何保证数据的最终一致性
一、背景 在微服务框架下,跨服务之间的调用,当遇到操作耗时或者量大的情况,我们一般会采用异步编程实现。 本文出现的问题是:异步回调过来时,却未查询到数据库中的任务,导致未能正常处理回调。 下面是当…...
vue3-dom-diff算法
vue3diff算法 什么是vue3diff算法 Vue3中的diff算法是一种用于比较虚拟DOM树之间差异的算法,其目的是为了高效地更新真实DOM,减少不必要的重渲染 主要过程 整个过程主要分为以下五步 前置预处理后置预处理仅处理新增仅处理后置处理包含新增、卸载、…...
年会抽奖Html
在这里插入图片描述 <!-- <video id"backgroundMusic" src"file:///D:/background.mp3" loop autoplay></video> --> <divstyle"width: 290px; height: 580px; margin-left: 20px; margin-top: 20px; background: url(D:/nianhu…...
ubuntu16 重启之后lvm信息丢失故障恢复
一、背景 1、问题背景 业务有一台物理开发服务器,文件系统有损坏;由于重启时没有检查,导致重启卡住。后面通过断电重新启动之后,无法进入系统;进入救援模式,注释数据盘挂载。重启之后进入系统,…...
【华为OD-E卷 - 热点网站统计 100分(python、java、c++、js、c)】
【华为OD-E卷 - 热点网站统计 100分(python、java、c、js、c)】 题目 企业路由器的统计页面,有一个功能需要动态统计公司访问最多的网页URL top N。请设计一个算法,可以高效动态统计Top N的页面 输入描述 每一行都是一个URL或…...
Ubuntu下安装Android Sdk
下载android sdk命令行工具 https://developer.android.com/studio?hlzh-cn#command-tools mkdir android-sdk cd android-sdk unzip commandlinetools-linux-11076708_latest.zip 添加环境变量到~/.bashrc export ANDROID_HOME$HOME/android-sdk export PATH$PATH:$ANDRO…...
【JVM】总结篇-类的加载篇之 类的加载器 和ClassLoader分析
文章目录 类的加载器ClassLoader自定义类加载器双亲委派机制概念源码分析优势劣势如何打破Tomcat 沙箱安全机制JDK9 双亲委派机制变化 类的加载器 获得当前类的ClassLoader clazz.getClassLoader() 获得当前线程上下文的ClassLoader Thread.currentThread().getContextClassLoa…...
怎样修改el-table主题样式
起因:el-table有主题样式,部分需要单独设置 环境:ideanodejs插件谷歌浏览器 第一步:找到scss文件: 谷歌浏览器打开表格页面,ctrlshifti打开开发者工具,点击后鼠标移动到表格单元格上单击一下…...
MySQL(二)MySQL DDL数据库定义语言
1. MySQL DDL数据库定义语言 1.1. MySQL定义语言 进入MySQL mysql -u root -p(回车后输入密码,即可进入mysq1)1.1.1. 数据库操作 (1)查看数据库 mysql>show databases;注:MySQL语句分隔符为“;” mysql库很重要它里面有…...
Spring Boot 项目启动报 NoClassDefFoundError 异常的原因分析与解决方案 - jackson 版本不一致
目录 报错: 问题分析: 解决方案: 方案 1:对 Jackson 版本进行统一 方案 2:升级 Springfox 版本 方案 3:替换 Springfox 为 springdoc-openapi(推荐) 方案 4:排除冲突的 Jack…...
原型与原型链
什么是原型(对象) 在JavaScript中,每个对象都具有一个原型对象prototype,目的是:利用原型对象实现在同一原型链中的原型方法共享 在理解原型对象前,需要先了解什么是构造函数 构造函数 用来初始化对象的…...
【Linux】信号处理
一、Linux系统信号 1、常见的系统信号 常见的Linux系统信号 信号值描述1SIGHUP挂起(hang up)进程2SIGINT中断进(interrupt)程3SIGQUIT停止(stop)进程9SIGKILL无条件终止(terminate)…...
5个不同类型的mysql数据库安装
各种社区版本下载官方地址:MySQL :: MySQL Community Downloads 一、在线YUM仓库(Linux) 选择 MySQL Yum Repository 选择对应版本下载仓库安装包(No thanks, just start my download.) 下载方法1:下载到本…...
python学习笔记—12—布尔类型、if语句
1. 布尔类型 (1) 定义 (2) 比较运算符 (3) 代码演示 1. 手动定义 bool_1 True bool_2 False print(f"bool_1的内容是:{bool_1}, 类型是:{type(bool_1)}") print(f"bool_2的内容是:{bool_2}, 类型是:{type(bool…...
分数阶傅里叶变换代码 MATLAB实现
function Faf myfrft(f, a) %分数阶傅里叶变换函数 %输入参数: %f:原始信号 %a:阶数 %输出结果: %原始信号的a阶傅里叶变换N length(f);%总采样点数 shft rem((0:N-1)fix(N/2),N)1;%此项等同于fftshift(1:N),起到翻…...
《数据结构》期末考试测试题【中】
《数据结构》期末考试测试题【中】 21.循环队列队空的判断条件为?22. 单链表的存储密度比1?23.单链表的那些操作的效率受链表长度的影响?24.顺序表中某元素的地址为?25.m叉树第K层的结点数为?26. 在双向循环链表某节点…...
openwrt 清缓存命令行
一、查看缓存 : free -m 二、清缓存:echo 3 > /proc/sys/vm/drop_caches 三、详解。 释放物理页缓存 echo 1 > /proc/sys/vm/drop_caches 释放可回收的slab对象,包含inode and dentry echo 2 > /proc/sys/vm/drop_caches 同时…...
RP2K:一个面向细粒度图像的大规模零售商品数据集
这是一种用于细粒度图像分类的新的大规模零售产品数据集。与以往专注于相对较少产品的数据集不同,我们收集了2000多种不同零售产品的35万张图像,这些图像直接在真实的零售商店的货架上拍摄。我们的数据集旨在推进零售对象识别的研究,该研究具…...
.NET Core FluentAPI
目录 约定配置 主要规则 两种配置方式 Data Annotation Fluent API Fluent API配置 Fluent API众多方法 选择 约定配置 主要规则 表名采用DbContext中的对应的DbSet的属性名。数据表列的名字采用实体类属性的名字,列的数据类型采用和实体类属性类型最兼容…...
【C++数据结构——查找】顺序查找(头歌实践教学平台习题)【合集】
目录😋 任务描述 相关知识 一、根据输入数据建立顺序表 二、顺序表的输出 三、顺序查找算法 测试说明 通关代码 测试结果 任务描述 本关任务:实现顺序查找的算法 相关知识 为了完成本关任务,你需要掌握: 根据输入数据建立…...
RTX 4090D+PyTorch 2.8实战:从零开始你的第一个AI项目
RTX 4090DPyTorch 2.8实战:从零开始你的第一个AI项目 1. 环境准备与快速验证 1.1 镜像优势解析 这个预装PyTorch 2.8的深度学习镜像专为RTX 4090D 24GB显卡优化,解决了AI开发者常见的三大痛点: 环境冲突:预装所有必要组件&…...
AgentCPM-Report深度应用:Pixel Epic智识终端多源数据整合研报生成
AgentCPM-Report深度应用:Pixel Epic智识终端多源数据整合研报生成 1. 产品概览:像素史诗智识终端 Pixel Epic智识终端是一款基于AgentCPM-Report大模型构建的创新研究报告生成系统。它将传统枯燥的科研分析过程转化为一场充满像素美学的数字冒险&…...
THE LEATHER ARCHIVE 快速上手指南:3步生成你的第一张AI时尚作品
THE LEATHER ARCHIVE 快速上手指南:3步生成你的第一张AI时尚作品 1. 项目简介 THE LEATHER ARCHIVE 是一款专为时尚设计师和创意人士打造的高端AI穿搭生成工具。它不同于传统的AI绘画工具,采用了独特的非对称剪贴报布局设计,让整个创作过程…...
面试官: 高并发系统概念解析(答案深度解析)持续更新
什么是高并发系统?——面试官想听的深度答案⚠️ 注意:“能扛住很多请求”不是高并发系统的定义,而是结果;面试官真正想考察的是你对“高并发本质”的理解、设计思维和落地经验。一、概念解释:别被字面意思带偏&#x…...
Chainlit+Qwen1.5-1.8B-GPTQ-Int4构建私有AI助手:支持文件上传与内容问答教程
ChainlitQwen1.5-1.8B-GPTQ-Int4构建私有AI助手:支持文件上传与内容问答教程 1. 学习目标与前置准备 今天我们来学习如何搭建一个功能强大的私有AI助手,这个助手不仅能进行智能对话,还能读取你上传的文件并回答相关问题。想象一下ÿ…...
Hive数据库入门指南:5分钟学会Flutter极速键值存储
Hive数据库入门指南:5分钟学会Flutter极速键值存储 【免费下载链接】hive Lightweight and blazing fast key-value database written in pure Dart. 项目地址: https://gitcode.com/gh_mirrors/hive/hive Hive是一款轻量级且速度极快的纯Dart键值数据库&…...
一个简洁易用的 Delphi JSON 封装库,基于 System.JSON`单元封装,提供更直观的 API廖
一、前言:什么是 OFA VQA 模型? OFA(One For All)是字节跳动提出的多模态预训练模型,支持视觉问答、图像描述、图像编辑等多种任务,其中视觉问答(VQA)是最常用的功能之一——输入一张…...
Llama-3.2V-11B-cot实战教程:从安装到图文问答,全程无报错操作手册
Llama-3.2V-11B-cot实战教程:从安装到图文问答,全程无报错操作手册 1. 工具简介 Llama-3.2V-11B-cot是一款基于Meta多模态大模型开发的高性能视觉推理工具,专门针对双卡4090环境进行了深度优化。这个工具最大的特点是解决了传统大模型部署中…...
lil_tea c++ style guide耸
一、中间件是啥?咱用“餐厅”打个比方 想象一下,你的FastAPI应用是个高级餐厅。 ?? 顾客(客户端请求)来到门口。- 迎宾(CORS中间件):先看你是不是从允许的街区(域名)来…...
022.模型评估指标:mAP、Precision、Recall、F1 Score的计算与解读
上周调一个YOLOv5的产线缺陷检测模型,测试集准确率看着挺高,上线后误报却把生产线搞停了。现场工程师抱怨:“你们这模型怎么乱报警?” 打开日志一看,模型把几个正常工件上的划痕阴影也框出来了——典型的精度不足问题。…...
