本人开发多线程qt5_c++工业上位机自动称重! Qt5之工业应用! 一套完整工程,工业电子称使用,无线扫码枪的使用,串口的使用,使用qt5.14,用qtcreator加载工程后,编译,运行,可调试可扩展,目的是学习qt开发方式。 不懂的随时和卖主沟通。 工业编程, 工业编程! 参数如下: ----------------------------- 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, 再点击重新构建 -----------------------------
工业上位机开发这活儿,总得有个趁手的工具箱。最近折腾的这套Qt5上位机自动称重系统,算是把工业场景里常见的坑都踩了一遍。项目里集成了电子秤数据采集、扫码枪通讯、多线程处理这些硬核功能,下面咱们边撸代码边唠嗑。
先看硬件交互这块硬骨头。电子秤通过串口传数据,得用QSerialPort配个心跳包:
// 串口初始化 QSerialPort* scalePort = new QSerialPort(this); scalePort->setPortName("COM3"); scalePort->setBaudRate(QSerialPort::Baud19200); if(!scalePort->open(QIODevice::ReadWrite)){ qCritical() << "电子秤连接失败,检查设备是否被占用"; } // 定时发送查询指令 QTimer* queryTimer = new QTimer(this); connect(queryTimer, &QTimer::timeout, [=](){ scalePort->write("W\r\n"); // 电子秤协议指令 }); queryTimer->start(500); // 每秒采集两次这里有个骚操作——用QTimer实现准实时采集,比纯事件驱动更可靠。注意数据解析时要处理粘包,上次就遇到个电子秤返回数据带换行符的坑。
扫码枪部分更刺激,得同时支持USB虚拟串口和蓝牙两种模式。这里用工厂模式封装:
class ScannerFactory { public: static AbstractScanner* createScanner(ScannerType type) { switch(type) { case USB_SCANNER: return new UsbHidScanner(); case BLUETOOTH_SCANNER: return new BluetoothScanner(QLowEnergyController::randomAddress()); default: throw std::invalid_argument("不支持的扫码枪类型"); } } };蓝牙低能耗(BLE)通讯这块,得注意Qt的QLowEnergyController在Windows下的玄学问题,有时候得手动重置蓝牙服务才能重连。
界面线程和硬件操作必须分开,上QThreadPool才稳当:
// 称重任务继承QRunnable class WeighingTask : public QRunnable { void run() override { QByteArray rawData = scalePort->readAll(); double weight = parseData(rawData); // 协议解析函数 QMetaObject::invokeMethod(qApp, [=](){ ui->weightLabel->setText(QString::number(weight, 'f', 1)+" kg"); }, Qt::QueuedConnection); } }; // 投递任务到线程池 QThreadPool::globalInstance()->start(new WeighingTask());这里有个关键点:UI更新必须用invokeMethod回到主线程,否则直接操作控件分分钟崩溃给你看。
数据库操作这块,用QMutex锁住写操作防止多线程踩踏:
QSqlDatabase db = QSqlDatabase::addDatabase("QODBC"); db.setDatabaseName("DRIVER={SQL Server};SERVER=192.168.1.100;DATABASE=WeighDB;"); QMutex dbMutex; void saveRecord(const WeighRecord &record) { QMutexLocker locker(&dbMutex); QSqlQuery query; query.prepare("INSERT INTO records (barcode, weight, timestamp) VALUES (?, ?, ?)"); query.addBindValue(record.barcode); query.addBindValue(record.weight); query.addBindValue(QDateTime::currentDateTime()); if(!query.exec()){ qWarning() << "入库失败:" << query.lastError().text(); } }注意Access数据库需要安装ODBC驱动,上次部署时忘了这个,现场调试差点被甲方打。
最后上点界面美化的私货,用qss搞个工业风:
/* style.qss */ QMainWindow { background: #2b2b2b; } QPushButton { background: #4CAF50; border-radius: 4px; padding: 8px; min-width: 80px; } QLabel#weightLabel { font: bold 28px 'Segoe UI'; color: #FF5722; qproperty-alignment: AlignCenter; }编译时切记关掉Shadow build,这玩意在混合调试时容易出幺蛾子。遇到链接错误先看是不是漏装了Windows SDK,再查pro文件里有没有忘加串口模块:
QT += core gui sql serialport CONFIG += c++11这套架子搭起来,后面加个Modbus/TCP或者OPC UA协议支持都不虚。工业软件嘛,核心就三字:稳、准、狠。数据采集要准,系统运行要稳,处理异常要狠——该丢包就丢包,别让整个系统卡死。有次现场遇到扫码枪频繁断连,直接写了个看门狗线程,超过3次失败就切备用设备,这才算镇住场子。