modbus 512 断线重连 db browser for sqlite
断线重连privateasyncTaskHeartbeatLoopAsync(CancellationTokentoken){// 监工一直循环干活直到工长喊停工token.IsCancellationRequestedwhile(!token.IsCancellationRequested){try{// 每隔一段时间检查一次最少200ms默认你设置的心跳间隔awaitTask.Delay(Math.Max(200,HeartbeatIntervalMs),token).ConfigureAwait(false);}catch(OperationCanceledException){break;// 工长喊停工监工立刻下班}// 拿一下工人最后一次成功读到数据的时间varlast_lastSuccessfulReadUtc;if(lastDateTime.MinValue)continue;// 还没读过跳过// 算一下现在距离最后一次成功读过了多久varageDateTime.UtcNow-last;// 如果没过超时时间 → 不管// 如果过了超时时间并且之前没报错过 → 判定【掉线了】if(!_heartbeatLostage.TotalMillisecondsHeartbeatTimeoutMs){_heartbeatLosttrue;// 标记掉线了try{OnHeartbeatLost?.Invoke();}catch{}// 通知外面掉线了// 如果开了【自动重连】→ 开始自救if(AutoReconnect){try{Disconnect();}catch{}// 先彻底断开清工人、清串口// 开始重试最多试 ReconnectMaxAttempts 次for(inti0;iReconnectMaxAttempts!token.IsCancellationRequested;i){try{// 重试间隔指数退避100ms → 200ms → 400ms → 800ms...Thread.Sleep(ReconnectBaseDelayMs*(int)Math.Pow(2,i));// 重新打开串口if(OpenSerial(_lastPortName,_lastBaudRate)){// 重建Modbus_modbusMasterModbusSerialMaster.CreateRtu(_serialPort);_modbusMaster.Transport.Retries1;_modbusMaster.Transport.WaitToRetryMilliseconds50;break;// 重连成功跳出循环}}catch{// 失败了不管继续重试}}}}}}awaitTask.Delay(Math.Max(200,HeartbeatIntervalMs),token).ConfigureAwait(false);我用最通俗、最扎心的大白话告诉你监工必须休息不休息你的电脑CPU会直接炸掉软件卡死动不了结合你这行代码逐点给你讲死awaitTask.Delay(Math.Max(200,HeartbeatIntervalMs),token).ConfigureAwait(false);1. 监工不休息会发生什么恐怖后果监工的工作是循环检查设备有没有掉线while(!token.IsCancellationRequested){// 检查掉线...}如果没有这行 Delay 休息监工会一秒钟循环几万次疯狂检查电脑CPU直接占满100%软件界面卡死、点不动设备发热、卡顿工业现场直接报废这就是死循环绝对不能允许2. 这行代码的真正作用让监工定时巡逻而不是疯跑Task.Delay(时间)监工歇一会儿再去检查不是偷懒是工业软件的保命写法比如你设置心跳间隔 1000ms → 监工歇1秒检查一次既不浪费CPU又能实时监控设备状态3. 为什么要加Math.Max(200, ...)双重保险防止你把心跳间隔设太小比如10ms强制最少休息200毫秒彻底杜绝监工疯跑、CPU爆满4. 为什么要带token监工休息的时候也能听见工会代表的停工命令一喊停立刻停止休息直接下班不耽误线程退出终极大白话总结监工休息 为了不浪费电脑资源不让软件卡死定时歇一会儿再去巡逻检查这是工业监控软件的标准规矩这行代码不是多余的是保命的这里的 token 是干嘛的停工保命用大白话翻译监工休息的时候手里必须攥着「停工令牌」核心作用监工正在休息Delay如果你关闭窗口、调用 Disconnect() → 工会代表_cts喊 停工这个 token 会立刻打断休息让监工马上停止循环、下班走人如果没有 token监工休息到一半你喊停工他听不见会一直赖在后台变成僵尸线程导致串口占用、软件卡死极简记忆token 监工休息时的「紧急停工按钮」二、.ConfigureAwait(false) 是干嘛的界面流畅用大白话翻译监工休息完就在后台自己干活别来打扰界面窗口核心作用工业软件必加你的软件有窗口界面主线程还有后台监工 / 工人子线程默认情况下后台任务休息完会跑回界面线程抢资源加了 .ConfigureAwait(false)监工全程在后台后台干活不抢界面资源窗口绝不卡顿、绝不假死极简记忆ConfigureAwait (false) 后台任务不打扰界面保证软件流畅不卡catch (OperationCanceledException) { break; }核心大白话翻译监工收到「强制停工通知」了立刻结束休息、退出循环、下班走人完整场景结合上面的 Delay 代码监工正在休息 await Task.Delay(…)你关闭窗口 / 调用 Disconnect() → 工会代表喊 Cancel() 停工手里的 token 立刻生效强行打断休息系统会抛出 OperationCanceledException取消操作异常catch 抓住这个异常 → 执行 break直接跳出监工的死循环监工彻底下班它是干嘛用的必须要写不写这行监工会因为被打断休息而报错崩溃写了这行优雅停工不报错、不残留、不占用串口资源工业软件必须这么写保证线程安全退出var age DateTime.UtcNow - last;计算从「工人最后一次成功读到数据」到「现在」一共过去了多长时间计算从「工人最后一次成功读到数据」到「现在」一共过去了多长时间_heartbeatLost作用标记设备是否已经掉线、是否已经触发过报警和重连age.TotalMilliseconds刚才算出来的 → 工人多久没读到数据了总毫秒数HeartbeatTimeoutMs你设定的 心跳超时时间比如 3000 毫秒 3 秒→ 这是底线超过这个时间没数据就算掉线工人摸鱼 / 断连的时间超过了我设定的最大容忍时间 → 判定设备掉线AutoReconnectif(AutoReconnect){try{Disconnect();}catch{}for(inti0;iReconnectMaxAttempts!token.IsCancellationRequested;i){try{Thread.Sleep(ReconnectBaseDelayMs*(int)Math.Pow(2,i));if(OpenSerial(_lastPortName,_lastBaudRate)){_modbusMasterModbusSerialMaster.CreateRtu(_serialPort);_modbusMaster.Transport.Retries1;_modbusMaster.Transport.WaitToRetryMilliseconds50;break;}}catch{}}1. if (AutoReconnect)→ 你允许监工自动救场吗允许 继续执行重连不允许 掉线就不管了2. try { Disconnect(); } catch { }→ 第一步彻底清场调用你写的满分 Disconnect()工会代表喊停工 → 工人 / 监工全部安全下班 → 关闭串口 → 销毁所有资源必须先清场才能重连防止线程打架、端口占用3. for (int i 0; i 最大重试次数 !token.IsCancellationRequested; i)→ 循环重试守两个规矩最多试 ReconnectMaxAttempts 次不无限死磕工会代表喊停工立刻停止重试不顽固4. Thread.Sleep(基础延迟 * 2^i)→ 指数退避重试工业核心第一次等 100ms第二次等 200ms第三次等 400ms第四次等 800ms…越重试等越久不疯狂占用 CPU不刷屏攻击串口5. OpenSerial(串口名, 波特率)→ 尝试重新打开串口工位打开成功 硬件恢复连接失败 继续下一次重试6. 重建 _modbusMaster→ 串口打开成功 → 拿新的 Modbus 通信工具配置好参数准备让新工人上岗读数据7. break→ 重连成功结束重试循环终极总结这套逻辑为什么稳先清场再重连绝对不线程冲突有限次数重试不卡死软件指数退避不浪费 CPU 资源随时响应停工命令安全退出全程 try-catch绝不崩溃
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2608489.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!