在我的工作中,可能打交道最多的就是串口通信了,與單片機(jī)進(jìn)行數(shù)據(jù)通信,串口無疑是最簡單的方式,今天我們使用 Qt 實(shí)現(xiàn)一個(gè)自己的串口調(diào)試助手。
實(shí)現(xiàn)目標(biāo)
自己編寫一個(gè)基于 Qt 的串口調(diào)試軟件,可以實(shí)現(xiàn)本軟件與串口助手之間的通訊。
軟件發(fā)送的數(shù)據(jù),經(jīng)虛擬串口轉(zhuǎn)發(fā),能夠在串口助手中正確接收;
串口助手發(fā)送的數(shù)據(jù)可以在本軟件的接收文本框中顯示,進(jìn)而實(shí)現(xiàn)串口數(shù)據(jù)雙向通信。
所需工具及環(huán)境
- 虛擬串口軟件(用于創(chuàng)建一對虛擬串口)Qt Creator 4.10.1Qt 5.13.1XCOM V2.0 串口助手本人電腦 Windows 10 64bit [版本 10.0.19041.329]
本文源碼
后臺(tái)回復(fù)關(guān)鍵字“Qt-COM”,獲取本文涉及到的虛擬串口軟件及 Qt 工程源碼。
界面設(shè)計(jì)
利用 Qt Creator 新建一個(gè) Project,模板選擇 Application--> Qt Widgets Application , 向?qū)е?Class Information 頁面中,Base class 選擇 QMainWindow 、 QWidget 、QDialog 都可以。
工程創(chuàng)建完畢,.ui 文件具體設(shè)計(jì)如下:
具體實(shí)現(xiàn)
導(dǎo)入串口通信模塊
從 Qt 5.1 版本開始,Qt 就有了自己的串口通訊類,之前版本需要使用第三方的串口通信類才行。
要想使用串口通信類,需要在 .pro 文件中添加 QT += serialport
顯示系統(tǒng)中所有串口號
顯示串口號列表的是一個(gè) QComboBox 控件。
我們調(diào)用 QSerialPortInfo::availablePorts() 可以獲得一個(gè) QList ,List 中的每一項(xiàng) QSerialPortInfo 代表一個(gè)串口實(shí)例,該類中保存了系統(tǒng)中已有串口的端口名稱、系統(tǒng)位置、描述和供應(yīng)商等信息。
遍歷系統(tǒng)中所有串口名的實(shí)現(xiàn)代碼如下:
QStringList MainWindow::getPortNameList()
{
QStringList m_serialPortName;
foreach(const QSerialPortInfo &info,QSerialPortInfo::availablePorts())
{
m_serialPortName << info.portName();
qDebug()<<"serialPortName:"< }
return m_serialPortName;
}
遍歷上面的 QList,將串口名稱保存至 m_serialPortName 變量中,這個(gè)變量的類型是 QStringList , 將最終結(jié)果顯示在 QComboBox 中:
m_portNameList = getPortNameList();
ui->comboBoxPortName->addItems(m_portNameList);
打開串口
串口的打開涉及到如下函數(shù):
// 判斷串口是否已打開
bool QIODevice::isOpen() const
// 清空緩沖區(qū)
bool QSerialPort::clear(QSerialPort::Directions directions = AllDirections)
// 串口關(guān)閉
[override virtual] void QSerialPort::close()
// 設(shè)置要打開的串口名
void QSerialPort::setPortName(const QString &name)
// 設(shè)置串口通信的波特率
bool QSerialPort::setBaudRate(qint32 baudRate, QSerialPort::Directions directions = AllDirections)
// 設(shè)置串口通信的數(shù)據(jù)位,數(shù)據(jù)位一般為 8 位
bool QSerialPort::setDataBits(QSerialPort::DataBits dataBits)
// 設(shè)置串口通信的流控制,一般無需流控制
bool QSerialPort::setFlowControl(QSerialPort::FlowControl flowControl)
// 設(shè)置串口通信的奇偶校驗(yàn),一般選擇“無”
bool QSerialPort::setParity(QSerialPort::Parity parity)
// 設(shè)置串口通信的停止位,停止位一般為 1
bool QSerialPort::setStopBits(QSerialPort::StopBits stopBits)
在“打開串口”按鈕上右鍵彈出菜單中,選擇 轉(zhuǎn)到槽 ... ,在按鍵的 clicked() 事件中,添加串口打開的對應(yīng)代碼。
串口通信類庫通信過程基本需要以下步驟,即:打開串口 --> 配置串口參數(shù)(波特率、數(shù)據(jù)位、停止位、奇偶校驗(yàn)、流控等) --> 收發(fā)數(shù)據(jù)。
串口打開的具體實(shí)現(xiàn)如下:
void MainWindow::on_btnOpenCOM_clicked()
{
if (ui->btnOpenCOM->text()=="打開串口")
{
if(m_serialPort->isOpen())
{
m_serialPort->clear();
m_serialPort->close();
}
m_serialPort->setPortName(m_portNameList[ui->comboBoxPortName->currentIndex()]);
if(!m_serialPort->open(QIODevice::ReadWrite))
{
qDebug()<comboBoxPortName->currentIndex()]<<"打開失敗!";
return;
}
// 打開成功
m_serialPort->setBaudRate(ui->comboBoxBaudRate->currentText().toInt(),QSerialPort::AllDirections);// 設(shè)置波特率和讀寫方向
m_serialPort->setDataBits(QSerialPort::Data8); // 數(shù)據(jù)位為 8 位
m_serialPort->setFlowControl(QSerialPort::NoFlowControl); // 無流控制
m_serialPort->setParity(QSerialPort::NoParity); // 無校驗(yàn)位
m_serialPort->setStopBits(QSerialPort::OneStop); // 一位停止位
connect(m_serialPort,SIGNAL(readyRead()),this,SLOT(receiveInfo()));
ui->btnOpenCOM->setText("關(guān)閉串口");
} else
{
m_serialPort->close();
ui->btnOpenCOM->setText("打開串口");
}
}
串口發(fā)送數(shù)據(jù)
串口發(fā)送數(shù)據(jù)的函數(shù)為:
qint64 QIODevice::write(const char *data)
這個(gè)函數(shù)是將以‘/0’結(jié)尾的字符串中的數(shù)據(jù)寫入設(shè)備(‘