物联网设备接入系统后如何查看硬件实时数据?
要在软件中实时查看硬件设备的信息,通常需要结合前后端技术来实现。以下是设计思路和实现步骤:
1. 系统架构设计
实时查看硬件设备信息的系统通常采用以下架构:
- 数据采集层: 硬件设备通过传感器采集数据,发送到InfluxDB。
- 数据存储层: InfluxDB存储设备的历史和实时数据。
- 后端服务层: 提供API接口,从InfluxDB查询数据并返回给前端。
- 前端展示层: 通过Web界面或移动端实时展示设备信息。
- 实时通信层: 使用WebSocket或Server-Sent Events (SSE)实现实时数据推送。
2. 实现步骤
(1) 数据采集与存储
硬件设备通过MQTT、HTTP或其他协议将数据发送到后端,后端将数据写入InfluxDB。可以参考前面的Java代码实现数据写入。
(2) 后端服务设计
后端需要提供API接口,用于查询设备的历史数据和实时数据。
API设计
-
查询历史数据: 返回设备在某个时间范围内的数据。
-
请求示例:
GET /api/devices/{deviceId}/history?start=2023-10-01T00:00:00Z&end=2023-10-02T00:00:00Z -
响应示例:
{"deviceId": "device_123","data": [{"time": "2023-10-01T12:00:00Z", "temperature": 25.3, "humidity": 60.1},{"time": "2023-10-01T12:05:00Z", "temperature": 25.5, "humidity": 60.0}] }
-
-
查询实时数据: 返回设备的最新数据。
-
请求示例:
GET /api/devices/{deviceId}/realtime -
响应示例:
{"deviceId": "device_123","time": "2023-10-01T12:10:00Z","temperature": 25.4,"humidity": 60.2 }
-
后端代码示例(Spring Boot + InfluxDB)
import com.influxdb.client.InfluxDBClient;
import com.influxdb.client.InfluxDBClientFactory;
import com.influxdb.client.QueryApi;
import com.influxdb.query.FluxRecord;
import com.influxdb.query.FluxTable;
import org.springframework.web.bind.annotation.*;import java.time.Instant;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;@RestController
@RequestMapping("/api/devices")
public class DeviceController {private final InfluxDBClient influxDBClient;public DeviceController() {// 初始化InfluxDB客户端this.influxDBClient = InfluxDBClientFactory.create("http://localhost:8086", "your-token".toCharArray(), "your-org", "your-bucket");}// 查询历史数据@GetMapping("/{deviceId}/history")public Map<String, Object> getHistoryData(@PathVariable String deviceId,@RequestParam String start,@RequestParam String end) {String fluxQuery = String.format("from(bucket: \"your-bucket\") " +"|> range(start: %s, stop: %s) " +"|> filter(fn: (r) => r._measurement == \"hardware_metrics\" and r.device_id == \"%s\")", start, end, deviceId);QueryApi queryApi = influxDBClient.getQueryApi();List<FluxTable> tables = queryApi.query(fluxQuery);List<Map<String, Object>> data = new ArrayList<>();for (FluxTable table : tables) {for (FluxRecord record : table.getRecords()) {Map<String, Object> point = new HashMap<>();point.put("time", record.getTime());point.put(record.getField(), record.getValue());data.add(point);}}Map<String, Object> response = new HashMap<>();response.put("deviceId", deviceId);response.put("data", data);return response;}// 查询实时数据@GetMapping("/{deviceId}/realtime")public Map<String, Object> getRealtimeData(@PathVariable String deviceId) {String fluxQuery = String.format("from(bucket: \"your-bucket\") " +"|> range(start: -1m) " + // 查询最近1分钟的数据"|> filter(fn: (r) => r._measurement == \"hardware_metrics\" and r.device_id == \"%s\") " +"|> last()", deviceId); // 获取最新的一条数据QueryApi queryApi = influxDBClient.getQueryApi();List<FluxTable> tables = queryApi.query(fluxQuery);Map<String, Object> response = new HashMap<>();if (!tables.isEmpty()) {FluxRecord record = tables.get(0).getRecords().get(0);response.put("deviceId", deviceId);response.put("time", record.getTime());response.put(record.getField(), record.getValue());}return response;}
}
(3) 前端实时展示
前端可以通过以下方式实现实时数据展示:
- 轮询(Polling): 定期调用后端API获取最新数据(简单但不高效)。
- WebSocket: 建立双向通信通道,后端主动推送数据到前端。
- Server-Sent Events (SSE): 后端单向推送数据到前端。
WebSocket实现示例
-
后端(Spring Boot):
import org.springframework.messaging.handler.annotation.MessageMapping; import org.springframework.messaging.handler.annotation.SendTo; import org.springframework.stereotype.Controller;@Controller public class WebSocketController {@MessageMapping("/device-data")@SendTo("/topic/device-data")public String sendDeviceData(String deviceId) {// 查询设备的最新数据并返回return "Device Data: " + deviceId;} } -
前端(JavaScript):
const socket = new WebSocket('ws://localhost:8080/ws'); socket.onmessage = function(event) {const data = JSON.parse(event.data);console.log("Received data:", data);// 更新UI };
SSE实现示例
-
后端(Spring Boot):
import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.PathVariable; import org.springframework.web.bind.annotation.RestController; import org.springframework.web.servlet.mvc.method.annotation.SseEmitter;@RestController public class SSEController {@GetMapping("/sse/device-data/{deviceId}")public SseEmitter streamDeviceData(@PathVariable String deviceId) {SseEmitter emitter = new SseEmitter();// 模拟实时数据推送new Thread(() -> {try {while (true) {String data = fetchDataFromInfluxDB(deviceId); // 从InfluxDB获取数据emitter.send(data);Thread.sleep(1000); // 每秒推送一次}} catch (Exception e) {emitter.completeWithError(e);}}).start();return emitter;}private String fetchDataFromInfluxDB(String deviceId) {// 查询InfluxDB并返回数据return "Device Data: " + deviceId;} } -
前端(JavaScript):
const eventSource = new EventSource('/sse/device-data/device_123'); eventSource.onmessage = function(event) {const data = JSON.parse(event.data);console.log("Received data:", data);// 更新UI };
(4) 前端UI设计
使用前端框架(如React、Vue.js)构建实时数据展示界面,例如:
- 实时数据卡片:显示设备的当前状态(温度、湿度等)。
- 历史数据图表:使用ECharts或Chart.js展示历史数据趋势。
3. 总结
通过InfluxDB存储设备数据,结合后端API和前端实时通信技术(如WebSocket或SSE),可以高效实现硬件设备信息的实时查看。关键点包括:
- 使用InfluxDB高效存储和查询时序数据。
- 后端提供API接口,支持历史数据和实时数据查询。
- 前端通过WebSocket或SSE实现实时数据推送和展示。
- 使用图表库展示历史数据趋势。
相关文章:
物联网设备接入系统后如何查看硬件实时数据?
要在软件中实时查看硬件设备的信息,通常需要结合前后端技术来实现。以下是设计思路和实现步骤: 1. 系统架构设计 实时查看硬件设备信息的系统通常采用以下架构: 数据采集层: 硬件设备通过传感器采集数据,发送到InfluxDB。数据存…...
最新版本TOMCAT+IntelliJ IDEA+MAVEN项目创建(JAVAWEB)
前期所需: 1.apache-tomcat-10.1.18-windows-x64(tomcat 10.1.8版本或者差不多新的版本都可以) 2.IntelliJ idea 24年版本 或更高版本 3.已经配置好MAVEN了(一定先配置MAVEN再搞TOMCAT会事半功倍很多) 如果有没配置…...
《生成对抗网络:当AI学会自我博弈的艺术》
2023年DALLE 2生成的《太空歌剧院》斩获艺术比赛大奖时,我在画作前驻足了整整十分钟——那些光影的渐变、笔触的韵律,竟来自两个神经网络的博弈游戏。这让我想起AlphaGo自我对弈突破人类棋谱局限的往事,生成对抗网络(GANÿ…...
【Linux学习笔记】Linux基本指令分析和权限的概念
【Linux学习笔记】Linux基本指令分析和权限的概念 🔥个人主页:大白的编程日记 🔥专栏:Linux学习笔记 文章目录 【Linux学习笔记】Linux基本指令分析和权限的概念前言一. 指令的分析1.1 alias 指令1.2 grep 指令1.3 zip/unzip 指…...
PHP之常用函数
在你有别的编程语言的基础下,你想学习PHP,可能要了解的一些关于常用函数的信息。 获取时间 //获取时间 后面跟自定义时间格式 echo date("Y-m-d H:i:s");删除变量 unset($a);判断变量是否存在 var_dump(isset($a));判断变量是否为null va…...
Leetcode 刷题记录 05 —— 普通数组
本系列为笔者的 Leetcode 刷题记录,顺序为 Hot 100 题官方顺序,根据标签命名,记录笔者总结的做题思路,附部分代码解释和疑问解答。 目录 01 最大子数组和 方法一:动态规划(卡达尼算法) 方法…...
【LLM】kimi 1.5模型架构和训练流程
note 推出两个多模态模型,深度思考模型 long-CoT 对标 o1,通用模型 short-CoT 模型对标 gpt-4o。 文章目录 note一、kimi 1.5模型训练流程预训练SFT训练long-CoT SFTRL训练long2short 小结Reference 一、kimi 1.5模型训练流程 推出两个多模态模型&…...
deepseek在pycharm中的配置和简单应用
对于最常用的调试python脚本开发环境pycharm,如何接入deepseek是我们窥探ai代码编写的第一步,熟悉起来总没坏处。 1、官网安装pycharm社区版(免费),如果需要安装专业版,需要另外找破解码。 2、安装Ollama…...
第二十四天 学习分布式数据管理,了解如何在多个设备间共享数据
HarmonyOS分布式数据管理实战:轻松实现多设备数据共享 一、为什么需要分布式数据管理? 在万物互联的时代,我们的智能设备数量正在快速增长。根据IDC最新报告,2023年平均每个用户拥有6.2台智能设备。HarmonyOS的分布式能力正是为…...
Android15 Camera框架中的StatusTracker
StatusTracker介绍 StatusTracker是Android15 Camera框架中用来协调Camera3各组件之间状态转换的类。 StatusTracker线程名:std::string("C3Dev-") mId "-Status" Camera3 StatusTracker工作原理 StatusTracker实现批处理(状态…...
MyBatis-Plus 注解大全
精心整理了最新的面试资料和简历模板,有需要的可以自行获取 点击前往百度网盘获取 点击前往夸克网盘获取 MyBatis-Plus 注解大全 MyBatis-Plus 是基于 MyBatis 的增强工具,通过注解简化了单表 CRUD 操作和复杂查询的配置。以下是常用注解的分类及详细说…...
【MySQL_03】数据库基本--核心概念
文章目录 一、数据库基础1.1 数据库基础定义1.2 数据库分类与典型产品1.3 数据库模型1.4 数据库层次结构1.5 数据库核心机制1.6 数据表和视图1.61 数据表(Table)1.62 视图(View) 1.7 键类型1.8 MySQL数据类型1.9 数据库范式化 二、…...
Ubuntu 下 nginx-1.24.0 源码分析 (1)
main 函数在 src\core\nginx.c int ngx_cdecl main(int argc, char *const *argv) {ngx_buf_t *b;ngx_log_t *log;ngx_uint_t i;ngx_cycle_t *cycle, init_cycle;ngx_conf_dump_t *cd;ngx_core_conf_t *ccf;ngx_debug_init(); 进入 main 函数 最…...
边缘计算盒子:解决交通拥堵的智能方案
在当今的智能交通系统中,边缘计算盒子(Edge Computing Box)正逐渐成为不可或缺的核心组件。这种设备通过将计算能力下沉到网络边缘,极大地提升了数据处理的速度和效率,特别适用于实时性要求极高的交通监控场景。本文将…...
工程化与框架系列(22)--前端性能优化(中)
前端性能优化(运行) 🏃 引言 运行时性能直接影响用户交互体验和应用流畅度。本文将深入探讨前端运行时性能优化的各种策略和技术,包括渲染优化、内存管理、计算优化等关键主题,帮助开发者构建高性能的Web应用。 运行…...
API调试工具的无解困境:白名单、动态IP与平台设计问题
引言 你是否曾经在开发中遇到过这样的尴尬情形:你打开了平台的API调试工具,准备一番操作,结果却发现根本无法连接到平台?别急,问题出在调试工具本身。今天我们要吐槽的就是那些神奇的开放平台API调试工具,…...
C#模拟鼠标点击,模拟鼠标双击,模拟鼠标恒定速度移动,可以看到轨迹
C#模拟鼠标点击,模拟鼠标双击,模拟鼠标恒定速度移动,可以看到轨迹 using System; using System.Collections.Generic; using System.Linq; using System.Runtime.InteropServices; using System.Text; using System.Threading.Tasks;namespa…...
php虚拟站点提示No input file specified时的问题及权限处理方法
访问站点,提示如下 No input file specified. 可能是文件权限有问题,也可能是“.user.ini”文件路径没有配置对,最简单的办法就是直接将它删除掉,还有就是将它设置正确 #配置成自己服务器上正确的路径 open_basedir/mnt/qiy/te…...
RISC-V汇编学习(三)—— RV指令集
有了前两节对于RISC-V汇编、寄存器、汇编语法等的认识,本节开始介绍RISC-V指令集和伪指令。 前面说了RISC-V的模块化特点,是以RV32I为作为ISA的核心模块,其他都是要基于此为基础,可以这样认为:RISC-V ISA 基本整数指…...
java 重点知识 — JVM存储模块与类加载器
1 jvm主要模块 方法区 存储了由类加载器从.class文件中解析的类的元数据(类型信息、域信息、方法信息)及运行时常量池(引用符号及字面量)。 所有线程共享;内存不要求连续,可扩展,可能发生垃圾回…...
WPF有哪些使用率高的框架
架构类库 Community Toolkit MVVMMVVM Light UI类库 MahApps.MetroMaterial Design In XAML Toolkit 图标类库 MahApps.Metro.IconPacks...
idea中使用DeepSeek让编程更加便捷
IDEA中使用DeepSeek让编程更加便捷 对于开发者来说,IDEA(IntelliJ IDEA)是一款强大的开发工具。但你是否知道,通过安装DeepSeek这款插件,可以让你的编程体验更上一层楼?今天,我们就来聊聊如何在…...
创建Electron35 + vue3 + electron-builder项目,有很过坑,记录过程
环境: node v20.18.0 npm 11.1.0 用到的所有依赖: "dependencies": {"core-js": "^3.8.3","vue": "^3.2.13","vue-router": "^4.5.0"},"devDependencies": {"ba…...
elasticsearch是哪家的
Elasticsearch:数据搜索与分析的领航者 在当今这个信息爆炸的时代,快速且准确地处理海量数据成为了众多企业和组织追求的目标。而Elasticsearch正是在这个背景下脱颖而出的一款强大的开源搜索引擎。它是由位于美国加利福尼亚州的Elastic公司所开发和维护…...
nginx基础http基础
目录 nginx简介正向代理&反向代理正向代理反向代理What Is a Reverse Proxy Server? High-Performance Load Balancing (负载均衡)Problem(问题)Solution(解决方案)常见负载均衡算法Round Robin(轮询)…...
5. MySQL 存储引擎(详解说明)
5. MySQL 存储引擎(详解说明) 文章目录 5. MySQL 存储引擎(详解说明)1. 查看存储引擎2. 设置系统默认的存储引擎3. 设置表的存储引擎3.1 创建表时指定存储引擎3.2 修改表的存储引擎 4. 引擎介绍4.1 InnoDB 引擎:具备外键支持功能的事务存储引擎4.2 MyISAM 引擎&…...
基于LabVIEW的伺服阀高频振动测试闭环控制系统
为实现伺服阀在设定位置上下快速移动(1kHz控制频率)的振动测试目标,需构建基于LabVIEW的闭环控制系统。系统需满足高速数据采集、实时控制算法(如PID或自适应控制)、高精度电流驱动及传感器反馈处理等需求。结合用户提…...
97.在 Vue 3 中使用 OpenLayers 根据两行根数 (TLE) 计算并显示卫星轨迹(EPSG:3857)
前言 在许多卫星应用场景中,我们需要 基于 TLE(Two-Line Element Set, 两行根数)计算卫星轨迹,并在地图上进行可视化。本文将使用 Vue 3 OpenLayers satellite.js,实现 实时计算卫星轨迹,并在地图上动态更…...
Android Coil总结
文章目录 Android Coil总结概述添加依赖用法基本用法占位图变形自定义ImageLoader取消加载协程支持缓存清除缓存监听 简单封装 Android Coil总结 概述 Coil 是一个用于 Android 的 Kotlin 图像加载库,旨在简化图像加载和显示的过程。它基于 Kotlin 协程࿰…...
fastjson漏洞#不出网#原理#流量特征
原理 本质是java的反序列化漏洞,由于引进了自动检测类型的(autotype)功能,fastjson在对json字符串反序列化的时候,会读取type内容,会试图将json内容反序列化成这个对象,并调用这个类的setter方…...
