当前位置: 首页 > article >正文

全国霸王餐 API 接口聚合平台,Java 后端多数据源路由策略设计

全国霸王餐 API 接口聚合平台Java 后端多数据源路由策略设计在构建全国性的霸王餐Free Meal与外卖CPS聚合平台时单一的数据源架构往往无法支撑海量的并发请求与复杂的业务隔离需求。随着业务规模的扩张系统通常面临多重挑战一方面需要对接美团、饿了么、抖音等多个第三方平台的异构API另一方面核心业务数据如用户信息、订单、佣金需要进行读写分离或分库分表处理。针对这些痛点本文将基于baodanbao.com.cn的架构演进深入探讨如何利用Spring的AbstractRoutingDataSource实现动态数据源路由以及如何设计一套灵活的注解驱动机制来支撑高可用的API聚合服务。一、 动态数据源路由核心AbstractRoutingDataSourceSpring提供的AbstractRoutingDataSource是实现多数据源切换的基石。其核心思想是通过重写determineCurrentLookupKey方法动态决定当前线程应使用哪个数据源Key。在霸王餐系统中我们需要根据当前的业务场景如查询用户余额使用主库查询商品列表使用从库或外部平台标识如美团数据源、饿了么数据源进行路由。定义数据源枚举首先我们需要定义系统中所有可能的数据源类型。这包括数据库的读写类型以及不同第三方平台的标识。packagebaodanbao.com.cn.enums;/** * 数据源类型枚举 * 定义系统支持的主从库及第三方平台源 * author baodanbao.com.cn */publicenumDataSourceType{/** * 主库 - 用于写操作 */MASTER,/** * 从库 - 用于读操作 */SLAVE,/** * 美团数据源 */MEITUAN,/** * 饿了么数据源 */ELEME,/** * 抖音数据源 */DOUYIN}数据源上下文持有者利用ThreadLocal存储当前线程的数据源标识保证线程安全。packagebaodanbao.com.cn.datasource;/** * 数据源上下文持有者 * 用于存储当前线程的数据源标识 * author baodanbao.com.cn */publicclassDynamicDataSourceContextHolder{privatestaticfinalThreadLocalStringCONTEXT_HOLDERnewThreadLocal();/** * 设置数据源 */publicstaticvoidsetDataSource(StringdataSourceKey){CONTEXT_HOLDER.set(dataSourceKey);}/** * 获取数据源 */publicstaticStringgetDataSource(){returnCONTEXT_HOLDER.get();}/** * 清除数据源 */publicstaticvoidclearDataSource(){CONTEXT_HOLDER.remove();}}动态数据源实现类继承AbstractRoutingDataSource实现路由逻辑。packagebaodanbao.com.cn.datasource;importorg.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;/** * 动态数据源实现 * 根据上下文决定使用哪个数据源 * author baodanbao.com.cn */publicclassDynamicDataSourceextendsAbstractRoutingDataSource{OverrideprotectedObjectdetermineCurrentLookupKey(){// 从上下文中获取数据源KeyStringdataSourceKeyDynamicDataSourceContextHolder.getDataSource();// 如果未指定默认使用主库returndataSourceKey!null?dataSourceKey:DataSourceType.MASTER.name();}}二、 注解驱动的数据源切换为了在Service层灵活控制数据源我们设计了DataSource注解。通过Spring AOP在方法执行前切换数据源。自定义注解允许在类或方法级别指定数据源类型。packagebaodanbao.com.cn.annotation;importbaodanbao.com.cn.enums.DataSourceType;importjava.lang.annotation.ElementType;importjava.lang.annotation.Retention;importjava.lang.annotation.RetentionPolicy;importjava.lang.annotation.Target;/** * 数据源注解 * 用于指定方法或类使用的数据源 * author baodanbao.com.cn */Target({ElementType.METHOD,ElementType.TYPE})Retention(RetentionPolicy.RUNTIME)publicinterfaceDataSource{DataSourceTypevalue()defaultDataSourceType.MASTER;}AOP 切面实现在方法执行前切换数据源执行后清除。packagebaodanbao.com.cn.aspect;importbaodanbao.com.cn.annotation.DataSource;importbaodanbao.com.cn.datasource.DynamicDataSourceContextHolder;importorg.aspectj.lang.ProceedingJoinPoint;importorg.aspectj.lang.annotation.Around;importorg.aspectj.lang.annotation.Aspect;importorg.aspectj.lang.annotation.Pointcut;importorg.aspectj.lang.reflect.MethodSignature;importorg.springframework.core.annotation.Order;importorg.springframework.stereotype.Component;importjava.lang.reflect.Method;/** * 数据源AOP切面 * 拦截DataSource注解实现动态切换 * author baodanbao.com.cn */AspectOrder(1)// 优先级需高于事务切面ComponentpublicclassDataSourceAspect{Pointcut(annotation(baodanbao.com.cn.annotation.DataSource) || within(baodanbao.com.cn.annotation.DataSource))publicvoiddataSourcePointCut(){}Around(dataSourcePointCut())publicObjectaround(ProceedingJoinPointpoint)throwsThrowable{// 获取目标方法MethodSignaturesignature(MethodSignature)point.getSignature();Methodmethodsignature.getMethod();// 获取类上的注解如果方法上没有则获取类上的DataSourcedataSourcemethod.getAnnotation(DataSource.class);if(dataSourcenull){dataSourcemethod.getDeclaringClass().getAnnotation(DataSource.class);}if(dataSource!null){// 设置数据源DynamicDataSourceContextHolder.setDataSource(dataSource.value().name());}try{// 执行方法returnpoint.proceed();}finally{// 清除数据源防止内存泄漏DynamicDataSourceContextHolder.clearDataSource();}}}三、 配置与实际业务应用配置类将多数据源配置到Spring容器中。packagebaodanbao.com.cn.config;importbaodanbao.com.cn.datasource.DynamicDataSource;importorg.springframework.boot.context.properties.ConfigurationProperties;importorg.springframework.boot.jdbc.DataSourceBuilder;importorg.springframework.context.annotation.Bean;importorg.springframework.context.annotation.Configuration;importorg.springframework.context.annotation.Primary;importjavax.sql.DataSource;importjava.util.HashMap;importjava.util.Map;/** * 数据源配置 * 初始化主库、从库及第三方平台数据源 * author baodanbao.com.cn */ConfigurationpublicclassDataSourceConfig{BeanConfigurationProperties(spring.datasource.master)publicDataSourcemasterDataSource(){returnDataSourceBuilder.create().build();}BeanConfigurationProperties(spring.datasource.slave)publicDataSourceslaveDataSource(){returnDataSourceBuilder.create().build();}BeanConfigurationProperties(spring.datasource.meituan)publicDataSourcemeituanDataSource(){returnDataSourceBuilder.create().build();}BeanPrimarypublicDataSourcedynamicDataSource(){DynamicDataSourcedynamicDataSourcenewDynamicDataSource();MapObject,ObjectdataSourceMapnewHashMap();dataSourceMap.put(master,masterDataSource());dataSourceMap.put(slave,slaveDataSource());dataSourceMap.put(meituan,meituanDataSource());// 可以继续添加饿了么、抖音等数据源...dynamicDataSource.setTargetDataSources(dataSourceMap);// 默认数据源dynamicDataSource.setDefaultTargetDataSource(masterDataSource());returndynamicDataSource;}}业务层应用在Service中使用注解指定数据源。packagebaodanbao.com.cn.service.impl;importbaodanbao.com.cn.annotation.DataSource;importbaodanbao.com.cn.enums.DataSourceType;importbaodanbao.com.cn.mapper.UserMapper;importbaodanbao.com.cn.service.UserService;importorg.springframework.beans.factory.annotation.Autowired;importorg.springframework.stereotype.Service;/** * 用户业务实现 * 演示读写分离与多源切换 * author baodanbao.com.cn */ServicepublicclassUserServiceImplimplementsUserService{AutowiredprivateUserMapperuserMapper;/** * 写操作插入用户强制使用主库 */OverrideDataSource(DataSourceType.MASTER)publicvoidcreateUser(StringuserId,Stringname){userMapper.insert(userId,name);}/** * 读操作查询用户信息使用从库 */OverrideDataSource(DataSourceType.SLAVE)publicUsergetUserById(StringuserId){returnuserMapper.selectById(userId);}/** * 跨平台业务调用美团数据源查询特定数据 * 假设我们有一个统一的接口规范底层通过数据源路由到美团的数据库或API配置 */DataSource(DataSourceType.MEITUAN)publicvoidsyncMeituanData(){// 这里可以是调用美团Mapper的方法// 框架会自动路由到meituan数据源System.out.println(当前线程数据源已切换至: DynamicDataSourceContextHolder.getDataSource());// ... 执行美团相关业务逻辑}}本文著作权归 俱美开放平台 转载请注明出处

相关文章:

全国霸王餐 API 接口聚合平台,Java 后端多数据源路由策略设计

全国霸王餐 API 接口聚合平台,Java 后端多数据源路由策略设计 在构建全国性的霸王餐(Free Meal)与外卖CPS聚合平台时,单一的数据源架构往往无法支撑海量的并发请求与复杂的业务隔离需求。随着业务规模的扩张,系统通常面…...

高可用外卖返利 CPS 平台:Java 后端异步回调处理机制深度解析

高可用外卖返利 CPS 平台:Java 后端异步回调处理机制深度解析 在构建外卖返利(CPS)系统时,异步回调(Callback)机制是连接用户授权、订单同步与佣金结算的神经中枢。美团、饿了么等平台的用户授权与订单状态…...

测试计划详细说明

一份高质量的测试计划本质上是质量风险的防御蓝图,它要在有限资源和无限质量诉求之间找到平衡点。我将从结构、内容、决策逻辑三个维度展开,并提供一个可直接落地的框架。一、测试计划的核心定位测试计划的本质回答三个问题:测什么&#xff1…...

当 95% 泳池拒绝轮椅人群时,“泳池升降机” 正在创造包容性蓝海​

在一座城市的游泳馆里,坐在轮椅上的小李望着眼前的泳池,眼神中满是渴望却又带着一丝无奈。以往,他只能看着别人在水中畅游,因为大部分泳池没有适合他这样行动不便者入水的设施。但最近,这家游泳馆引入了一款便携式泳池…...

嵌入式系统高可靠存储模块(Store)设计与实现

在嵌入式系统开发中,数据持久化是保障设备稳定性、连续性运行的核心支撑,尤其对于物联网网关、工业控制器、智能家居终端等设备,需长期存储设备配置、运行参数、网络信息等关键数据,且要求掉电不丢失、读写高效、容错性强。基于此…...

CPU与操作系统【简单的认识理解】

在日常开发过程中,我们都是正常写完代码去执行即可,不用了解计算机运行的底层逻辑。但是了解计算机运行的底层逻辑,对于我们以后理解撰写代码以及理解错误原理有着重要地位,因此,我们特意写上一篇文章跟大家介绍。在计…...

Agno 多 Agent 实战(二):搭建完整内容创作流水线

前情回顾 上一篇我们用路由模式做了一个智能问答系统,路由模式适合单步任务,一次分配。今天分享的是更复杂的场景:多步骤协作。 很多任务不是一步能做完的,比如写一篇文章:得先找资料,再写初稿,再审核修改,最后排版。这就需要多个 Agent 一步步协作,我们可以用协调模…...

2025_NIPS_G1: Teaching LLMs to Reason on Graphs with Reinforcement Learning

文章核心总结与创新点 核心内容 本文针对大型语言模型(LLMs)在图推理任务中表现有限的问题,提出了一种基于强化学习(RL)的方法G1。通过在大规模合成图论任务数据集Erdős上训练,G1显著提升了LLMs的图推理能力,且在未见过的任务、领域和图编码方案中表现出强泛化性,同…...

MySql(简单处理查询结果--查询结果去重)

3. 现在运营需要查看用户来自于哪些学校,请从用户信息表中取出学校的去重数据。示例:user_profileiddevice_idgenderageuniversityprovince12138male21北京大学Beijing23214male复旦大学Shanghai36543female20北京大学Beijing42315female23浙江大学ZheJiang55432mal…...

AllCells细胞原料解析:Leukopak与PBMC在CGT中的应用【曼博生物供应人原代细胞】

AllCells细胞原料体系解析:Leukopak与PBMC在CGT中的应用 摘要: AllCells作为DLS体系中的重要品牌,提供GMP与RUO级人源细胞原料,包括Leukopak与PBMC等产品类型,广泛应用于细胞与基因治疗研发及生产流程。 关键词&#x…...

02-LangChain简单介绍、RAG开发

一、LangCain1、介绍LangChain由Harrison Chase创建于2022年10月,它是围绕LLMs(大语言模型)建立的一个框架。LangChain自身并不开发LLMs,它的核心理念是为各种LLMs实现通用的接口,把LLMs相关的组件“链接”在一起&…...

软考高项-第六章-项目管理概论

项目和项目集重点在于正确的做事,项目组合在于做正确的事组织过程资产:过程资产,治理文件,数据资产,知识资产,安保和安全事业环境因素:市场条件,社会和文化影响因素,监管…...

惯性导航解算及误差分析

目录 1.连续时间下三维运动的微分性质 1.1 旋转矩阵的微分方程 1.2 四元数的微分方程 1.3 旋转向量的微分方程 2.惯性导航解算 2.1 姿态更新 2.2 速度更新 2.3 位置更新 3.惯性导航误差分析 3.1 姿态误差微分方程 3.2 速度误差微分方程 3.3 位置误差方程 3.4 bias…...

手把手教程:快速设置远程开机,看完就会

今天就给大家带来一份完整、可直接照着操作的远程开机教程,即可实现无需公网 IP、一键远程唤醒,随时随地让设备为你待命。设备支持检查确认主板支持WAKE-ON-LAN(网络唤醒)功能,局域网内需具备两台设备:目标…...

资深大模型工程师详细讲解:RAG召回率优化三重微调实战

✅ 一、核心策略再解构:从“三层次”到“五维协同链路”原有“数据-索引-查询”三层结构非常精准,但为了更贴近企业级复杂场景,我们进一步抽象为 五维协同链路:维度关键目标是否可微调微调切入点1. 数据生成质量构建高质量正负样本…...

关系型数据库星型模型聚合表生成

在关系型数据库(MySQL、Oracle、SQL Server等)中,通过星型模型模拟多维分析结构,高效生成聚合表,解决报表查询慢、多维分析繁琐、实时计算压力大等核心痛点。 一、前置基础 星型模型是关系型数据库模拟多维结构的最优方…...

GNSS导航信号模拟器 卫星导航定位模拟器 GNSS卫星导航定位信号模拟器行业应用解决方案 GNSS模拟器

随着全球卫星导航系统的全面建设与深度应用,各类卫星导航定位授时终端已广泛渗透到交通、物联网、通信、测绘、消费电子等众多领域。但在终端产品的研发、测试、量产全流程中,行业长期面临诸多核心痛点:传统外场实地测试模式需投入大量人力物…...

Java 基础核心知识

文章目录1. 谈谈对AQS的理解2. fail-safe机制与fail-fast机制分别有什么作用3. new String("abc")到底创建了几个对象4. 对序列化和反序列化的理解5. 谈谈对Java中SPI的理解6. String、StringBuffer、StringBuilder区别7. Integer 的判断8. 深拷贝和浅拷贝9. 强引用、…...

csp预习day2

set#include<bits/stdc.h> using namespace std;int main(){// ios::sync_with_stdio(0);// cin.tie(0);// cout.tie(0);int n,m; //值域、询问个数scanf("%d%d",&n,&m);int set[n1]; //大小为n的随机序列for (int i 1; i < n; i){scanf(&qu…...

ARM Cortex M0 and M0+ 学习:Architecture

Block Diagram Operation Mode The ARMv6-M architecture has two operation modes and two states. In addition, it can have privileged and unprivileged access levels. Core Registers R0-R12:通用寄存器 R13(SP):存储主栈指针MSP或进程指针PSP,目的是帮助CPU在栈中…...

Ltspice-线性电流控制电流源F/电压源H

上一篇我们聊了功能强大的任意行为源&#xff08;BV/BI&#xff09;&#xff0c;它们像是一个可以编写任意公式的“万能计算器”。而在实际电路中&#xff0c;还有一类更基础、更经典的元件&#xff0c;它们遵循严格的线性比例关系&#xff0c;这就是我们今天要介绍的线性受控源…...

黑马点评-“附近商户“功能无法实现

问题分析 由于Redis版本不够&#xff0c;因此我们使用不了GEOSEARCH命令(Redis 6.2.0 或更高版本) 需要升级Redis版本 下载高版本Redis Redis8.6.2点此下载 后缀的区别: 带有 -with-Service vs 不带 with-Service&#xff1a; 包含了将 Redis 注册为 Windows 系统后台服务的相关…...

【架构师通关】理发店排队 + 车库停车,大白话秒懂“进程状态模型”与“PV操作

兄弟们&#xff0c;操作系统的进程管理一直是软考里最让人头疼的“硬骨头” &#x1f9b4;。什么“阻塞”、“挂起”、“信号量”、“PV操作”&#xff0c;听着就像天书 &#x1f4da;。 但今天&#xff0c;飞哥绝不跟你拽学术名词&#xff01;咱们就通过“去理发店剪个头” &a…...

如何在 React Native 中使用 Expo AV 高效缓存视频文件

本文详解在裸 React Native 项目中集成 expo-av 实现视频本地缓存的完整方案&#xff0c;涵盖路径处理、文件写入、URI 适配及常见兼容性问题&#xff08;如 react-native-fs 路径不被 expo-av 识别&#xff09;&#xff0c;并提供可直接运行的优化代码与关键注意事项。 本…...

mysql备份工具选择_mysqldump对InnoDB与MyISAM支持

mysqldump默认对MyISAM用表级锁、InnoDB不启用事务快照&#xff0c;混合引擎必须用--lock-all-tables保证一致性&#xff0c;且需确保REPEATABLE READ隔离级别和ROW/MIXED binlog格式。mysqldump 默认行为对 InnoDB 和 MyISAM 完全不同默认不加任何参数时&#xff0c;mysqldump…...

Go语言如何部署到K8s_Go语言Kubernetes部署教程【进阶】

Go服务容器化失败主因是镜像路径与WORKDIR不匹配、containerPort未对齐监听端口、Probe未适配程序健康接口、ConfigMap/Secret挂载权限不足&#xff0c;需逐一核验镜像内容、网络声明、文件权限及进程监听行为。Go 服务打包成容器镜像时&#xff0c;main.go 路径和 WORKDIR 不匹…...

c++如何通过文件映射mmap在多进程间实现高性能数据共享【进阶】

mmap 多进程共享必须用 MAP_SHARED&#xff0c;因其确保所有进程映射同一物理页并同步回文件&#xff1b;MAP_PRIVATE 为写时复制&#xff0c;修改不共享。需 O_RDWR 打开、ftruncate 预设大小&#xff0c;并配合适当同步机制。为什么 mmap 在多进程共享中必须用 MAP_SHARED 而…...

宝塔面板怎样实现数据库的多地异地自动备份_结合阿里云OSS与定时任务插件

宝塔面板需通过定时任务ossutilmysqldump实现阿里云OSS数据库自动备份&#xff1a;先配置ossutil及MySQL凭据文件&#xff0c;再编写含时间戳命名与NTP校时的Shell脚本&#xff0c;避免依赖无效的远程备份模块。宝塔面板怎么配置阿里云OSS自动备份数据库能&#xff0c;但得绕过…...

OpenAI估值逼近6万亿!连散户都杀入,但天价融资背后的“算计”,让人细思极恐

出品 | 网易智能 作者 | 小小 编辑 | 王凤枝 1220亿美元&#xff08;约合8800亿元人民币&#xff09;的承诺资金&#xff0c;8520亿美元&#xff08;约合6.1万亿元人民币&#xff09;的投后估值。 3月31日&#xff0c;OpenAI正式官宣了一轮规模惊人的融资&#xff0c;甚至连散户…...

【DIY小记】解决MacOS上Edge浏览器bilibili全屏卡顿的问题

近日笔者发现自己Macbook-Pro播放B站视频&#xff0c;全屏的时候必然卡顿&#xff0c;退出全屏就没事。笔者电脑的参数是&#xff1a; 芯片&#xff1a;M3系统&#xff1a;Tahoe 26.4浏览器&#xff1a;Edge 到网上一查发现《Edge浏览器在MacOS 26(Tahoe)系统上看B站卡顿》一…...