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

SpringBoot+Mybatis多数据源实战:TDengine与MySQL混搭的物联网数据存储方案

SpringBootMybatis多数据源实战TDengine与MySQL混搭的物联网数据存储方案在物联网系统开发中数据存储架构的设计往往面临一个核心矛盾海量设备时序数据的高效存储与业务数据的复杂关系处理如何平衡传统单一数据库方案要么在时序数据处理上捉襟见肘要么无法满足业务系统的灵活查询需求。本文将揭示如何通过SpringBootMybatis构建多数据源系统让TDengine与MySQL各司其职实现112的存储效果。1. 物联网数据存储的架构抉择当单个传感器每天产生上千条数据时十万级设备规模的系统很快就会面临单表亿级数据的处理压力。我们曾在一个智慧农业项目中实测发现MySQL单表存储传感器数据达到3000万条时按设备ID时间范围查询响应时间超过8秒同样的查询条件在TDengine中执行仅需23毫秒但TDengine在处理用户权限、设备元数据等关联查询时性能反而不及MySQL这种差异源于两种数据库的设计哲学特性MySQLTDengine数据模型关系型时序型索引效率B树索引适合点查询时间线索引适合范围扫描压缩比一般2-3倍典型10-20倍分布式支持需中间件原生支持事务完整性ACID完备最终一致性提示时序数据库并非要替代关系型数据库而是专为时间序列数据优化的特殊形态2. SpringBoot多数据源的核心配置实现双数据库协同工作的关键在于正确的依赖配置和数据源隔离。以下是经过生产验证的POM配置要点!-- 基础依赖 -- dependency groupIdorg.springframework.boot/groupId artifactIdspring-boot-starter-jdbc/artifactId /dependency dependency groupIdorg.mybatis.spring.boot/groupId artifactIdmybatis-spring-boot-starter/artifactId version2.2.0/version /dependency !-- 数据库驱动 -- dependency groupIdmysql/groupId artifactIdmysql-connector-java/artifactId scoperuntime/scope /dependency dependency groupIdcom.taosdata.jdbc/groupId artifactIdtaos-jdbcdriver/artifactId version3.2.8/version /dependency !-- 分页插件 -- dependency groupIdcom.github.pagehelper/groupId artifactIdpagehelper-spring-boot-starter/artifactId version1.4.1/version /dependency应用配置文件中需要明确区分两个数据源# MySQL数据源 spring.datasource.mysql.jdbc-urljdbc:mysql://host:3306/iot_biz?useSSLfalse spring.datasource.mysql.usernameroot spring.datasource.mysql.password123456 spring.datasource.mysql.driver-class-namecom.mysql.cj.jdbc.Driver # TDengine数据源 spring.datasource.taos.jdbc-urljdbc:TAOS://host:6030/iot_tsdb spring.datasource.taos.usernameroot spring.datasource.taos.passwordtaosdata spring.datasource.taos.driver-class-namecom.taosdata.jdbc.TSDBDriver3. Mybatis的多数据源隔离策略实现真正的数据源隔离需要三个层面的配合3.1 数据源配置类MySQL主数据源配置带Primary注解Configuration MapperScan(basePackages com.iot.dao.mysql, sqlSessionFactoryRef mysqlSqlSessionFactory) public class MysqlDataSourceConfig { Bean Primary ConfigurationProperties(spring.datasource.mysql) public DataSource mysqlDataSource() { return DataSourceBuilder.create().build(); } Bean Primary public SqlSessionFactory mysqlSqlSessionFactory( Qualifier(mysqlDataSource) DataSource dataSource) throws Exception { SqlSessionFactoryBean bean new SqlSessionFactoryBean(); bean.setDataSource(dataSource); bean.setMapperLocations(new PathMatchingResourcePatternResolver() .getResources(classpath:mapper/mysql/*.xml)); return bean.getObject(); } Bean Primary public DataSourceTransactionManager mysqlTransactionManager( Qualifier(mysqlDataSource) DataSource dataSource) { return new DataSourceTransactionManager(dataSource); } }TDengine从数据源配置Configuration MapperScan(basePackages com.iot.dao.taos, sqlSessionFactoryRef taosSqlSessionFactory) public class TaosDataSourceConfig { Bean ConfigurationProperties(spring.datasource.taos) public DataSource taosDataSource() { return DataSourceBuilder.create().build(); } Bean public SqlSessionFactory taosSqlSessionFactory( Qualifier(taosDataSource) DataSource dataSource) throws Exception { SqlSessionFactoryBean bean new SqlSessionFactoryBean(); bean.setDataSource(dataSource); bean.setMapperLocations(new PathMatchingResourcePatternResolver() .getResources(classpath:mapper/taos/*.xml)); return bean.getObject(); } Bean public DataSourceTransactionManager taosTransactionManager( Qualifier(taosDataSource) DataSource dataSource) { return new DataSourceTransactionManager(dataSource); } }3.2 Mapper接口与XML的物理隔离建议采用以下项目结构src/main/java └── com.iot.dao ├── mysql │ ├── DeviceMapper.java │ └── UserMapper.java └── taos ├── SensorDataMapper.java └── AlertLogMapper.java src/main/resources └── mapper ├── mysql │ ├── DeviceMapper.xml │ └── UserMapper.xml └── taos ├── SensorDataMapper.xml └── AlertLogMapper.xml3.3 事务管理的明确指定在Service层必须显式声明使用的事务管理器Service public class HybridService { // MySQL操作使用主事务管理器 Transactional(transactionManager mysqlTransactionManager) public void updateDeviceInfo(Device device) { deviceMapper.update(device); } // TDengine操作使用从事务管理器 Transactional(transactionManager taosTransactionManager) public void batchInsertSensorData(ListSensorData dataList) { sensorDataMapper.batchInsert(dataList); } // 跨库事务需采用最终一致性模式 public void processDeviceAlert(Device device, Alert alert) { updateDeviceStatus(device); // MySQL操作 saveAlertLog(alert); // TDengine操作 // 这里不能保证原子性需要业务层补偿机制 } }4. 性能优化实战技巧4.1 TDengine的表设计策略针对物联网场景的典型优化方案超级表子表模式CREATE STABLE IF NOT EXISTS sensors ( ts TIMESTAMP, temperature FLOAT, humidity FLOAT ) TAGS (device_id BINARY(64), region BINARY(32)); -- 每个设备自动建子表 INSERT INTO device_001 USING sensors TAGS (device_001, east) VALUES (now, 25.3, 60.2);批量写入优化Repository public interface SensorDataMapper { Insert({ script, INSERT INTO ${tableName} VALUES, foreach collectionlist itemitem separator,, (#{item.ts}, #{item.value}, #{item.status}), /foreach, /script }) void batchInsert(Param(tableName) String tableName, Param(list) ListSensorData data); }查询性能对比测试环境单节点TDengine 3.0设备数量10万数据总量15亿条查询类型响应时间(MySQL)响应时间(TDengine)单设备最近24小时数据3200ms28ms设备组7天统计报表超时(30s)420ms全系统1小时异常检测不可行680ms4.2 MySQL业务查询优化对于必须保留在MySQL的业务数据建议为设备元数据表添加复合索引ALTER TABLE device_info ADD INDEX idx_region_status (region, status);使用覆盖索引减少回表Select(SELECT device_id, device_name FROM device_info WHERE region #{region}) ListDeviceSimple listByRegion(String region);分页查询优化方案select idselectDevicePage resultTypeDevice SELECT * FROM device_info WHERE id #{lastId} ORDER BY id ASC LIMIT #{pageSize} /select5. 混合查询的优雅实现当需要同时关联业务属性与时序数据时可采用以下模式public ListDeviceDetail getDeviceDashboard(String deviceId) { // 从MySQL获取设备元数据 DeviceMeta meta deviceMapper.selectById(deviceId); // 从TDengine获取实时监测数据 SensorData data sensorDataMapper.selectLatest(deviceId); // 组合业务对象 return DeviceDetail.builder() .deviceId(deviceId) .deviceName(meta.getName()) .location(meta.getLocation()) .currentTemp(data.getTemperature()) .currentHumidity(data.getHumidity()) .status(calculateStatus(data)) .build(); }对于需要复杂关联的场景可以考虑物化视图模式定期将TDengine的统计结果同步到MySQL缓存层加速用Redis缓存热点设备的混合查询结果异步预处理通过消息队列触发后台计算任务在最近的一个智慧楼宇项目中这种混合架构帮助我们将系统响应时间从平均4.7秒降低到380毫秒同时存储成本下降60%。某能源监控平台实施后单服务器最高支持了12万台设备的实时数据采集95%的查询响应时间在1秒内完成。

相关文章:

SpringBoot+Mybatis多数据源实战:TDengine与MySQL混搭的物联网数据存储方案

SpringBootMybatis多数据源实战:TDengine与MySQL混搭的物联网数据存储方案 在物联网系统开发中,数据存储架构的设计往往面临一个核心矛盾:海量设备时序数据的高效存储与业务数据的复杂关系处理如何平衡?传统单一数据库方案要么在时…...

模型安全实践:实时手机检测-通用输入图像异常检测(模糊/过曝/裁剪)

模型安全实践:实时手机检测-通用输入图像异常检测(模糊/过曝/裁剪) 1. 项目简介与核心价值 在日常的手机检测应用中,我们经常会遇到各种图像质量问题:图片模糊看不清手机细节、光线过曝导致手机轮廓丢失、或者图片被…...

Stable-Diffusion-V1-5 效率工具集:Ollama本地LLM辅助提示词生成

Stable-Diffusion-V1-5 效率工具集:Ollama本地LLM辅助提示词生成 你是不是也遇到过这种情况:脑子里有个绝妙的画面,但打开Stable Diffusion,面对那个空白的提示词输入框,却不知道从何写起。要么写得太简单&#xff0c…...

别再手动调参了!用C#和Halcon的HSmartWindow控件,5分钟搞定ROI绘制与参数提取

工业视觉开发革命:用C#封装Halcon ROI的智能实践 在半导体检测、精密零件测量等工业场景中,区域兴趣(ROI)的精准定义直接影响着算法效果。传统开发模式下,工程师需要反复在Halcon脚本与C#界面代码间切换,手…...

【C++ 面试突击 · 07】大厂高频面试题:从菱形继承到const与constexpr的博弈深度解析

目录 1. 什么是菱形继承?怎么解决菱形继承? 2. 如何定义一个只能在堆上(栈上)生成对象的类? 3. C 强制类型转换运算符有哪些? 4. C 中的类型推导(auto)是如何工作的?…...

Loop:Mac窗口管理的优雅革命,开源免费的全新体验

Loop:Mac窗口管理的优雅革命,开源免费的全新体验 【免费下载链接】Loop MacOS窗口管理 项目地址: https://gitcode.com/GitHub_Trending/lo/Loop 你是否曾在多窗口工作中迷失方向?Loop作为一款开源的macOS窗口管理工具,通过…...

破解除密!ncmdumpGUI让你的NCM音乐文件重获自由

破解除密!ncmdumpGUI让你的NCM音乐文件重获自由 【免费下载链接】ncmdumpGUI C#版本网易云音乐ncm文件格式转换,Windows图形界面版本 项目地址: https://gitcode.com/gh_mirrors/nc/ncmdumpGUI 当你精心收藏的数百首网易云音乐歌曲,在…...

别再手动复制粘贴了!用CubeMX一键生成FreeRTOS工程(STM32F4 HAL库实战)

告别繁琐配置:STM32CubeMXFreeRTOS全自动工程生成指南 在嵌入式开发领域,时间就是竞争力。传统FreeRTOS移植需要手动复制文件、配置路径、修改中断向量表,稍有不慎就会陷入头文件缺失、链接错误的泥潭。现在,STM32CubeMX的图形化…...

ModelNet数据集高效下载与预处理实战指南

1. ModelNet数据集简介与下载技巧 ModelNet数据集是三维计算机视觉领域的经典基准数据集,由麻省理工学院CSAIL实验室于2015年发布。这个数据集最初是为了解决三维形状分类和检索问题而创建的,如今已成为点云处理、三维重建等研究的标配测试平台。 数据…...

AUTOSAR CANFM模块中,BusOff恢复的50ms和1000ms周期到底怎么来的?底层驱动配置详解

AUTOSAR CANFM模块中BusOff恢复时序的硬件级解析 在车载ECU开发中,CAN总线通信的可靠性直接关系到整车功能安全。当节点因连续错误进入BusOff状态时,AUTOSAR标准定义的50ms快恢复周期和1000ms慢恢复周期并非随意设定,而是源于CAN控制器硬件特…...

如何用d2s-editor高效管理暗黑破坏神2存档:终极可视化编辑指南

如何用d2s-editor高效管理暗黑破坏神2存档:终极可视化编辑指南 【免费下载链接】d2s-editor 项目地址: https://gitcode.com/gh_mirrors/d2/d2s-editor d2s-editor是一款免费开源的Web版暗黑破坏神2存档编辑器,它将复杂的二进制存档文件转化为直…...

AI Agent驱动业务规则测试:从复杂逻辑到精准用例的自动化实践

1. AI Agent如何重塑业务规则测试 第一次接触AI Agent驱动的测试用例生成时,我正被一个保险理赔系统的测试工作折磨得焦头烂额。那套系统里有上百条复杂的业务规则,光是理解"投保人年龄超过60岁且保单满5年但未达10年时,赔付比例调整为8…...

Electron-builder打包Windows应用,我踩过的三个坑(附详细解决方案)

Electron-builder打包Windows应用:三个典型问题的深度解析与实战解决方案 第一次使用electron-builder打包Windows应用时,那种期待与焦虑交织的感觉至今记忆犹新。作为一个从Web前端转向桌面应用开发的程序员,我本以为有了Electron这个跨平台…...

ncmdumpGUI终极指南:解锁你的音乐收藏,告别NCM格式束缚

ncmdumpGUI终极指南:解锁你的音乐收藏,告别NCM格式束缚 【免费下载链接】ncmdumpGUI C#版本网易云音乐ncm文件格式转换,Windows图形界面版本 项目地址: https://gitcode.com/gh_mirrors/nc/ncmdumpGUI 你是否曾经遇到过这样的情况&am…...

手把手教你用NEWLab搭建智能温控系统(附完整代码)

手把手教你用NEWLab搭建智能温控系统(附完整代码) 在智能家居和工业自动化领域,温度控制始终是核心需求之一。无论是保持室内舒适环境,还是确保精密设备的稳定运行,一套可靠的温控系统都不可或缺。对于物联网初学者和…...

FreeRTOS中断管理实战:如何用信号量优雅处理硬件中断(附STM32代码)

FreeRTOS中断管理实战:信号量在STM32硬件中断中的高效应用 1. 嵌入式实时系统中的中断挑战 在嵌入式开发中,中断处理就像餐厅里的紧急订单——它可能随时打断主厨正在准备的常规菜品。想象你正在安静地享用下午茶,突然门铃响起(…...

【仅限首批内测用户开放】Polars 2.0清洗性能调优白皮书:含12个未公开API、3类CPU亲和性绑定策略

第一章:Polars 2.0大规模数据清洗技巧概览Polars 2.0 在性能、内存效率与API一致性上实现重大升级,为TB级结构化数据清洗提供了低延迟、高吞吐的原生解决方案。其基于Arrow 15的列式引擎、零拷贝切片能力及多线程LazyFrame执行计划优化,使复杂…...

DreamScene2动态桌面软件:为Windows桌面注入活力的终极解决方案

DreamScene2动态桌面软件:为Windows桌面注入活力的终极解决方案 【免费下载链接】DreamScene2 一个小而快并且功能强大的 Windows 动态桌面软件 项目地址: https://gitcode.com/gh_mirrors/dr/DreamScene2 厌倦了千篇一律的静态桌面背景吗?DreamS…...

从51到STM32:手把手教你用STM32CubeMX和PWM驱动智能小车电机(附代码避坑)

从51到STM32:智能小车电机控制的进阶实战指南 十年前用51单片机做智能小车时,PWM配置需要手动计算定时器重装载值,而今天在STM32CubeMX里勾选几下就能生成精准的PWM信号——这就像从手动挡升级到了自动驾驶。作为过来人,我完整记…...

生物信息学入门:手把手教你用Java实现Needleman-Wunsch序列比对算法

生物信息学实战:用Java构建Needleman-Wunsch全局序列比对工具 第一次接触DNA序列比对时,看着两条看似杂乱无章的碱基序列在算法处理后突然呈现出惊人的相似性,那种发现隐藏规律的震撼感至今难忘。作为生物信息学领域最经典的算法之一&#xf…...

手机号与QQ号关联查询工具:技术原理与实战指南

手机号与QQ号关联查询工具:技术原理与实战指南 【免费下载链接】phone2qq 项目地址: https://gitcode.com/gh_mirrors/ph/phone2qq 破解数字身份关联难题:phone2qq工具的价值定位 在多账号管理场景中,用户经常面临数字身份关联断层问…...

GitHub自动化神器:用Cursor+Firecrawl实现项目自更新(避坑指南)

GitHub自动化神器:用CursorFirecrawl实现项目自更新(避坑指南) 在开源项目的日常维护中,重复性的更新工作往往消耗开发者大量精力。有没有一种方法,能让项目像拥有自我意识般自动完成内容搜集、代码生成甚至PR提交&am…...

如何完美解决MacBook触控板在Windows的三指拖动难题

如何完美解决MacBook触控板在Windows的三指拖动难题 【免费下载链接】ThreeFingersDragOnWindows Enables macOS-style three-finger dragging functionality on Windows Precision touchpads. 项目地址: https://gitcode.com/gh_mirrors/th/ThreeFingersDragOnWindows …...

Ubuntu服务器上配置KVM虚拟化环境:从零搭建Windows开发环境

1. 为什么要在Ubuntu服务器上跑Windows? 很多开发者可能都有这样的困惑:明明手头有性能强劲的Ubuntu服务器,但某些开发工具只能在Windows环境下运行。比如Visual Studio、SQL Server Management Studio这些微软系工具,或者某些行业…...

使用Java实现数据的生产和消费

【Kafka】Java实现数据的生产和消费 Kafka介绍 Kafka 是由 LinkedIn 公司开发的,它是一个分布式的,支持多分区、多副本,基于 Zookeeper 的分布式消息流平台,它同时也是一款开源的基于发布订阅模式的消息引擎系统。 Kafka术语 …...

智能AI识别之集装箱缺陷识别 集装箱数据集 集装箱缺陷数据集 集装箱凹陷数据集 集装箱锈蚀孔洞图像数据集 yolo数据集地10624期

📦 集装箱缺陷检测计算机视觉模型( 这是一个基于 YOLOv8/YOLOv10 框架的工业级目标检测模型,专门用于识别集装箱表面的三类典型缺陷。🔍 核心信息 模型类型:目标检测(Object Detection)基础框架…...

3D Face HRN真实案例:用于司法鉴定中面部特征三维比对辅助系统

3D Face HRN真实案例:用于司法鉴定中面部特征三维比对辅助系统 1. 引言:从平面照片到三维证据的突破 在司法鉴定领域,面部特征比对一直是身份识别的重要技术手段。传统的2D照片比对方法存在角度、光照、表情等多重限制,往往难以…...

革新性暗黑破坏神2存档管理开源工具:d2s-editor全功能解析

革新性暗黑破坏神2存档管理开源工具:d2s-editor全功能解析 【免费下载链接】d2s-editor 项目地址: https://gitcode.com/gh_mirrors/d2/d2s-editor 暗黑破坏神2存档修改门槛高?复杂二进制格式难以操作?d2s-editor作为免费开源的Web端…...

Java面试高频:阿里真实面试题——Redis分布式锁实现(3分钟速通,不会直接挂)

一、真实面试场景(代入感拉满) 上周,一个候选人来面试阿里P6。 技术面已经过了两轮,表现都不错。 最后一轮,面试官只问了一个问题: “你们项目里用过Redis分布式锁吗?怎么实现的?…...

Windows下用CMake和MinGW编译NLopt 2.6.2的完整指南(附测试代码)

Windows平台下NLopt 2.6.2源码编译与实战应用全解析 在科学计算与工程优化领域,NLopt作为一款开源的非线性优化库,因其丰富的算法支持和跨平台特性而广受欢迎。本文将深入探讨如何在Windows系统中从零开始构建NLopt 2.6.2开发环境,并通过完整…...