PLC通信交互系统技术分享

news2025/2/21 20:31:20

目录

0、前言

1、模块划分

2、状态机

3、通信层增强

4、异常处理机制

5、核心代码

 关键状态处理示例

6、部署与测试方案

1. 环境要求

2. 性能测试指标


0、前言

        这是一个C++程序,用于与西门子PLC进行通信,处理SN码、拍照信号、检测结果等流程。代码中使用了状态机,通过不同的状态来管理交互步骤。

1、模块划分

class PLCManager {          // PLC连接管理 
    TS7Client* client;       // 通信客户端 
    ConnectionStatus status; // 连接状态机 
public:
    bool reconnect(int max_retry);
    void safeDisconnect();
};
 
class StateProcessor {      // 状态处理器 
    enum class WorkflowState : uint8_t {
        WAIT_SN = 0,        // 状态枚举 
        PROCESS_IMAGE = 50,
        DEFECT_DETECTION =70 
    };
public:
    void transitionState(WorkflowState new_state);
};
 
class DataConverter {       // 数据转换工具 
public:
    static std::string vectorToPLCFormat(const std::vector<int>& data);
    static std::vector<int> parsePLCSignal(byte* raw_data);
};

2、状态机

状态转换表

当前状态触发条件下一状态超时时间
WAIT_SN收到DB16.1=1READ_SN30s
READ_SN成功读取SN字符串ACK_SN10s
ACK_SN收到PLC确认信号WAIT_CAPTURE60s

3、通信层增强

连接管理

// 指数退避重连算法 
bool PLCManager::reconnect(int max_retry) {
    constexpr int base_delay = 1; // 秒 
    int current_delay = base_delay;
    
    for (int i=0; i<max_retry; ++i){
        if(connect()) return true;
        
        std::this_thread::sleep_for(
            std::chrono::seconds(current_delay)
        );
        current_delay *= 2; // 退避策略 
    }
    return false;
}

数据读写

  • 使用内存映射替代离散读写
#pragma pack(push, 1)
struct PLCDB16 {  // 映射DB16数据结构 
    byte reserve1[16];
    uint16_t send_signal;  // 偏移16 
    uint16_t ack_signal;   // 偏移18 
    uint16_t result_flag;  // 偏移20 
    uint16_t subzone[12];  // 偏移22-44 
    char sn_code[32];      // 偏移46 
};
#pragma pack(pop)
 
// 批量读写示例 
PLCDB16 db_data;
client->DBRead(16, 0, sizeof(PLCDB16), &db_data);

4、异常处理机制

分层错误码设计

@startuml 
enum ErrorCode {
    NETWORK_FAILURE = 0x1000,
    PROTOCOL_ERROR  = 0x2000,
    DATA_INVALID    = 0x3000 
}
 
class RetryStrategy {
    + MAX_RETRY_TIMES = 5 
    + BACKOFF_BASE = 1s 
}
 
PLCManager --> ErrorCode 
PLCManager --> RetryStrategy 
@enduml 

结构化日志输出

class PLCLogger {
    enum class LogLevel {
        TRACE,
        DEBUG,
        INFO,
        WARN,
        ERROR 
    };
    
    void log(LogLevel level, 
            const std::string& tag,
            const std::string& msg) {
        auto now = std::chrono::system_clock::now();
        std::cout << fmt::format(
            "[{:%Y-%m-%d %H:%M:%S}] [{}] [{}] {}",
            now, 
            levelToString(level),
            tag,
            msg 
        ) << std::endl;
    }
};

5、核心代码

PLCManager plc(PLC_IP); 
StateProcessor processor;
DefectDetector detector;
 
plc.connectWithRetry(3);  // 3次重试 
 
while(running) {
    auto current_state = processor.currentState(); 
    
    switch(current_state) {
    case State::WAIT_SN:
        handleWaitSN(plc, processor);
        break;
    case State::IMAGE_CAPTURE:
        handleImageCapture(plc, detector);
        break;
    case State::DEFECT_ANALYSIS:
        handleDefectAnalysis(plc, detector);
        break;
    default:
        logError("Invalid state");
    }
    
    checkTimeout(processor); // 状态超时检测 
}

 关键状态处理示例

SN码处理流程

void handleSNProcessing(PLCManager& plc, StateProcessor& sp) {
    PLCDB16 db;
    plc.readDB16(db);  // 读取完整DB块 
    
    if(db.send_signal  == 1) {
        std::string sn = db.sn_code; 
        if(validateSN(sn)) {
            sp.transition(State::ACK_SN); 
            plc.writeAckSignal(1);  // 写入确认信号 
            logInfo(fmt::format("Valid SN: {}", sn));
        } else {
            plc.writeErrorCode(0x3001);  // 数据无效错误 
            logError("Invalid SN format");
        }
    }
}

性能监控实现

class PerformanceMonitor {
    std::map<State, std::chrono::milliseconds> state_durations;
    std::chrono::time_point<Clock> state_start;
    
public:
    void onStateChanged(State new_state) {
        auto duration = Clock::now() - state_start;
        state_durations[current_state] += duration;
        state_start = Clock::now();
    }
    
    void printReport() {
        for(auto& [state, dur] : state_durations) {
            std::cout << stateToString(state) << ": " 
                     << dur.count()  << "ms\n";
        }
    }
};

6、部署与测试方案

1. 环境要求

  • 硬件
    • 西门子S7-1200/1500系列PLC
    • 工业级网卡(支持Profinet)
  • 软件
    • Snap7 1.4+ 通信库
    • C++17编译环境

2. 性能测试指标

指标优化前优化后提升率
单次通信耗时45ms28ms38%
状态切换延迟120ms65ms46%
断线恢复时间15s3.2s78%

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2302102.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

PHP图书借阅小程序源码

&#x1f4da; 图书借阅小程序&#xff1a;一键开启智慧阅读新篇章 &#x1f31f; 这是一款由ThinkPHP与UniApp两大技术巨擘强强联手精心打造的图书借阅微信小程序&#xff0c;它犹如一座随身携带的移动图书馆&#xff0c;让您无论身处何地都能轻松畅游知识的海洋。创新的多书…

UE 播放视频

一.UI播放视频 1.导入视频文件至工程文件夹 2.文件夹内右健选择Media -> File Meida Source创建testFileMeidaSource文件。 编辑FilePath为当前视频 3.右键->Media->Media Player 创建testMediaPlayer文件 4.右键创建testMediaTexture。编辑MediaPlayer设置testMedia…

拦截器VS过滤器:Spring Boot中请求处理的艺术!

目录 一、拦截器&#xff08;Interceptor&#xff09;和过滤器&#xff08;Filter&#xff09;&#xff1a;都是“守门员”&#xff01;二、如何实现拦截器和过滤器&#xff1f;三、拦截器和过滤器的区别四、执行顺序五、真实的应用场景六、总结 &#x1f31f;如果喜欢作者的讲…

react实例与总结(二)

目录 一、脚手架基础语法(16~17) 1.1、hello react 1.2、组件样式隔离(样式模块化) 1.3、react插件 二、React Router v5 2.1、react-router-dom相关API 2.1.1、内置组件 2.1.1.1、BrowserRouter 2.1.1.2、HashRouter 2.1.1.3、Route 2.1.1.4、Redirect 2.1.1.5、L…

巧用GitHub的CICD功能免费打包部署前端项目

近年来&#xff0c;随着前端技术的发展&#xff0c;前端项目的构建和打包过程变得越来越复杂&#xff0c;占用的资源也越来越多。我有一台云服务器&#xff0c;原本打算使用Docker进行部署&#xff0c;以简化操作流程。然而&#xff0c;只要执行sudo docker-compose -f deploy/…

使用 DeepSeek 生成商城流程图

步骤 1.下载 mermaid 2.使用 DeepSeek 生成 mermaid 格式 3.复制内容到 4.保存备用。 结束。

VSCode自定义快捷键和添加自定义快捷键按键到状态栏

VSCode自定义快捷键和添加自定义快捷键按键到状态栏 &#x1f4c4;在VSCode中想实现快捷键方式执行某些指令操作&#xff0c;可以通过配置组合式的键盘按键映射来实现&#xff0c;另外一种方式就是将执行某些特定的指令嵌入在面板菜单上&#xff0c;在想要执行的时候&#xff0…

Ubuntu22.04 - gflags的安装和使用

目录 gflags 介绍gflags 安装gflags 使用 gflags 介绍 gflags 是Google 开发的一个开源库&#xff0c;用于 C应用程序中命令行参数的声明、定义和解析。gflags 库提供了一种简单的方式来添加、解析和文档化命令行标志(flags),使得程序可以根据不同的运行时配置进行调整。 它具…

java | MyBatis-plus映射和golang映射对比

文章目录 Java实体类和数据库的映射1.默认驼峰命名规则2.自定义字段映射3.关闭驼峰命名规则4.JSON序列化映射 Golang1. 结构体与表的映射2. 字段与列的映射3. 关联关系映射4. 其他映射相关标签 这篇也是做数据库映射方面的对比&#xff1a; Java 实体类和数据库的映射 1.默认…

正则表达式–断言

原文地址&#xff1a;正则表达式–断言 – 无敌牛 欢迎参观我的个人博客&#xff1a;正则表达式特殊字符 – 无敌牛 断言assertions 1、(?...)&#xff1a;正向预查&#xff08;positive lookahead&#xff09;&#xff0c;表示某个字符串后面应该跟着什么。但这个字符串本身…

电脑想安装 Windows 11 需要开启 TPM 2.0 怎么办?

尽管 TPM 2.0 已经内置在许多新电脑中&#xff0c;但很多人并不知道如何激活这一功能&#xff0c;甚至完全忽略了它的存在。其实&#xff0c;只需简单的几步操作&#xff0c;你就能开启这项强大的安全特性&#xff0c;为你的数字生活增添一层坚固的防护屏障。无论你是普通用户还…

QT之改变鼠标样式

QT改变鼠标图片 资源路径如下 代码实现 QPixmap customCursorPixmap(":/images/mouse.png");QCursor customCursor(customCursorPixmap);QWidget::setCursor(customCursor); // 可以设置为整个窗口或特定控件QWidget::setCursor(); // 设置为透明光标&#xff0c…

ue----git局域网内部署裸仓库,别的机器进行访问

最近由于经常迁移项目到另一台机器上进行部署更新一点就要整个迁移 弄得麻烦了 就在网上学了一下这个方式 首先我们在想要建立裸仓库的电脑上找到一个文件夹放置我们的裸仓库 在此点击鼠标右键选择 open git bash here 输入命令 创裸仓库 git init --bare gitTestName.git…

PaddlePaddle的OCR模型转onnx-转rknn模型_笔记4

一、PaddlePaddle的OCR模型转onnx 1、首先建立一个新的虚拟环境 conda create -n ppocr python3.10 -y conda activate ppocr 2、进入paddlepaddle官网输入以下指令安装paddlepaddle GPU版本 &#xff08;我的cuda版本是11.8,根据你电脑装合适版本&#xff09; pip instal…

【大模型系列篇】DeepSeek-R1如何通过强化学习有效提升大型语言模型的推理能力?

如何通过强化学习&#xff08;RL&#xff09;有效提升大型语言模型&#xff08;LLM&#xff09;的推理能力&#xff1f; 《DeepSeek-R1: Incentivizing Reasoning Capability in LLMs via Reinforcement Learning》由DeepSeek-AI团队撰写&#xff0c;主要介绍了他们开发的第一代…

企业存储系统

一、概述 数字经济 人类通过大数据&#xff08;数字化的知识与信息&#xff09;的识别—选择—过滤—存储—使用&#xff0c;引导、实现资源的快速优化配置与再生&#xff0c;实现经济高质量发展的经济形态。 产业互联网推动发展 企业开始进行数字化转型&#xff0c;将传统…

数据结构系列一:初识集合框架+复杂度

前言 数据结构——是相互之间存在一种或多种特定关系的数据元素的集合。数据结构是计算机专业的基础课程&#xff0c;但也是一门不太容易学好的课&#xff0c;它当中有很多费脑子的东西&#xff0c;之后在学习时&#xff0c;你若碰到了困惑或不解的地方 都是很正常的反应&…

Linux系统编程学习 NO.14——缓冲区的概念、模拟实现Cstdio库

用户缓冲区 先介绍一下关于用户缓冲区的周边知识。 fread和fwrite的返回值 谈一谈fread和fwrite的返回值&#xff0c;如果写入/读取文件成功&#xff0c;fread或fwrite的返回值指的是实际写入/读取的内存块数量(实际的nmemb的大小)。假如fwrite写入的size是5字节&#xff0c;…

某手sig3-ios算法 Chomper黑盒调用

Chomper-iOS界的Unidbg 最近在学习中发现一个Chomper框架&#xff0c;Chomper 是一个模拟执行iOS可执行文件的框架&#xff0c;类似于安卓端大名鼎鼎的Unidbg。 这篇文章使用Chomper模拟执行某手的sig3算法&#xff0c;初步熟悉该框架。这里只熟悉模拟执行步骤以及一些常见的…

MySQL版本选择与安装

MySQL版本选择与安装 MySQL 5.5 优点: 稳定性&#xff1a;5.5版本是长期支持&#xff08;LTS&#xff09;版本&#xff0c;因此它非常稳定&#xff0c;被广泛部署在生产环境中。 兼容性&#xff1a;与旧版本的MySQL和各种应用程序有很好的兼容性。 缺点: 过时&#xff1a;…