QT_C++多线程生产制造MES 1,现场实战项目。 2,这是一个汽车部件制造企业的一条厂线现场精密控制。 3,由本人单独完成。 设计技术众多,C++,PLC,OPC,工业以太网(扫码枪),串口扫码枪,多种数据库(多台设备)无缝连接与切换。 与该公司内部MES无缝链接。 4,提供yd码! 工业编程! 工业编程 参数如下: ----------------------------- 1)编程语言:\\t\\tC++ (11或以上); ----------------------------- 2)编程环境:\\t\\tQT5.14; ----------------------------- 3)编程工具1:\\t\\tqss ; ----------------------------- 4)编译器:\\t\\tmsvc ;(没有就完整安装2019,一定要选msvc,或 \\t\\t\\t安装 WIN10 SDK) ----------------------------- 5)数据库:\\t\\taccess, mysql, sqlserver ; ----------------------------- 6)如何加载pro文件\\t文件->打开文件或项目; \\t\\t\\t在Build&Run 下选择 Qt 5.14.2 msvc2017(或2015) \\t\\t\\t左侧边栏点击项目,右边概要下的Shadow build 不用勾选; ----------------------------- 7) SDK\\t\\t\\t需要安装win10SDK(编译或调试要用) ----------------------------- 8) 构建\\t\\t\\t记得先qtmake, 再点击重新构建 -----------------------------

在汽车零部件车间里,十几个PLC控制着机械臂精准走位,我盯着产线看板突然跳红——某个工位的扫码枪数据断流了。这可不是大学作业,产线停摆每分钟损失四位数。摸出包里焊着三防胶的工控机,打开那个用QT写的MES监控程序,今天聊聊怎么用C++在工业修罗场里活下来。

多线程不是炫技是保命

流水线有6台扫码设备(3台TCP/IP、3台RS232),数据必须实时入库。直接在主线程里轮询?等着被车间主任用扳手追杀吧。看看线程池的核心代码:

// 扫码线程基类
class ScanThread : public QObject {
    Q_OBJECT
public:
    explicit ScanThread(QObject *parent = nullptr) : QObject(parent) {}
    virtual void startListen() = 0;
signals:
    void dataReceived(const QByteArray &data);
};

// 处理TCP扫码枪的孙子类
class EthernetScanner : public ScanThread {
public:
    void startListen() override {
        server.listen(QHostAddress::Any, 9500);
        connect(&server, &QTcpServer::newConnection, [=](){
            while(auto socket = server.nextPendingConnection()) {
                connect(socket, &QTcpSocket::readyRead, [=](){
                    auto data = socket->readAll();
                    emit dataReceived(data.mid(3, 10)); // 截取有效报文段
                    socket->disconnectFromHost();
                });
            }
        });
    }
private:
    QTcpServer server;
};

每个设备类型继承ScanThread,通过信号槽与主线程交互。重点在数据包的截取——工业协议往往有固定报文头,用mid切分比正则快10倍。

数据库切换的暗战

QT_C++多线程生产制造MES 1,现场实战项目。 2,这是一个汽车部件制造企业的一条厂线现场精密控制。 3,由本人单独完成。 设计技术众多,C++,PLC,OPC,工业以太网(扫码枪),串口扫码枪,多种数据库(多台设备)无缝连接与切换。 与该公司内部MES无缝链接。 4,提供yd码! 工业编程! 工业编程 参数如下: ----------------------------- 1)编程语言:\\t\\tC++ (11或以上); ----------------------------- 2)编程环境:\\t\\tQT5.14; ----------------------------- 3)编程工具1:\\t\\tqss ; ----------------------------- 4)编译器:\\t\\tmsvc ;(没有就完整安装2019,一定要选msvc,或 \\t\\t\\t安装 WIN10 SDK) ----------------------------- 5)数据库:\\t\\taccess, mysql, sqlserver ; ----------------------------- 6)如何加载pro文件\\t文件->打开文件或项目; \\t\\t\\t在Build&Run 下选择 Qt 5.14.2 msvc2017(或2015) \\t\\t\\t左侧边栏点击项目,右边概要下的Shadow build 不用勾选; ----------------------------- 7) SDK\\t\\t\\t需要安装win10SDK(编译或调试要用) ----------------------------- 8) 构建\\t\\t\\t记得先qtmake, 再点击重新构建 -----------------------------

产线同时用着Access记录日志、MySQL存工艺参数、SQLServer对接集团MES。当年用ODBC通用接口差点翻车,后来改成动态加载驱动:

bool DBManager::switchDatabase(DatabaseType type) {
    QSqlDatabase db = QSqlDatabase::addDatabase(getDriverName(type), "active_conn");
    db.setHostName(Config::getHost(type));
    //...其他参数
    if (!db.open()) {
        qCritical() << "DB dead:" << db.lastError().text();
        emergencySaveToLocal(); // 本地存txt保命
        return false;
    }
    return true;
}

QString DBManager::getDriverName(DatabaseType type) const {
    static QMap<DatabaseType, QString> driverMap = {
        {Access, "QODBC"},
        {MySQL, "QMYSQL3"},
        {SqlServer, "QODBC"}
    };
    return driverMap.value(type);
}

遇到过MSSQL的ODBC驱动在Win7工控机上突然抽风,后来发现是系统时间不同步导致SSL握手失败——工业现场没有NTP服务器,只能写个定时校时线程。

硬件通信的玄学时刻

PLC通过OPC DA协议传输状态数据,最头疼的是数据类型转换。某个阀门的压力值在C++里显示正常,存入数据库就变负数。最终发现是Modbus的32位浮点处理问题:

// 处理4字节寄存器转float
float parsePLCData(const QByteArray &data) {
    if(data.size() < 4) return 0.0f;
    
    uint32_t raw = (static_cast<uint32_t>(data[0]) << 24) |
                   (static_cast<uint32_t>(data[1]) << 16) |
                   (static_cast<uint32_t>(data[2]) << 8)  |
                   static_cast<uint32_t>(data[3]);
                   
    // 处理PLC端的大小端问题
    if(isBigEndian_) {
        raw = qFromBigEndian(raw);
    } else {
        raw = qFromLittleEndian(raw);
    }
    
    return *reinterpret_cast<float*>(&raw);
}

这代码拯救了当时因为数据异常导致整批零件报废的事故。工业编程的残酷在于,一个位运算失误可能让企业损失几十万。

最后给同行们的血泪经验

  1. Shadow build千万别勾!工控机硬盘速度堪比U盘,构建时遇到过增量编译失败导致程序闪退
  2. 多线程同步用QMutex不如QReadWriteLock,扫码枪这类高频但数据量小的场景,读锁比写锁快47%
  3. 数据库连接务必放在独立线程,曾经因为主线程SQL阻塞导致界面卡死,被操作工误认为是死机猛拍屏幕
  4. 在pro文件里加上win32: LIBS += -lwinmm,解决某些工控机没有高精度定时器的问题

凌晨三点的车间,设备嗡鸣声中和BUG搏斗的身影,才是工业软件工程师的日常。代码不仅要能跑,更要扛得住油污、电磁干扰、还有操作大叔的佛山无影脚。(测试过某台工控机接键盘口有漏电,遂在代码里加了防误触双确认弹窗——这都是课本里不会写的生存技能)

Logo

腾讯云面向开发者汇聚海量精品云计算使用和开发经验,营造开放的云计算技术生态圈。

更多推荐