加入星計劃,您可以享受以下權(quán)益:

  • 創(chuàng)作內(nèi)容快速變現(xiàn)
  • 行業(yè)影響力擴散
  • 作品版權(quán)保護
  • 300W+ 專業(yè)用戶
  • 1.5W+ 優(yōu)質(zhì)創(chuàng)作者
  • 5000+ 長期合作伙伴
立即加入
  • 正文
  • 相關(guān)推薦
  • 電子產(chǎn)業(yè)圖譜
申請入駐 產(chǎn)業(yè)圖譜

使用Qt打造屬于自己的串口調(diào)試助手

2021/01/18
885
  • 1評論
閱讀需 4 分鐘
加入交流群
掃碼加入
獲取工程師必備禮包
參與熱點資訊討論

在我的工作中,可能打交道最多的就是串口通信了,與單片機進行數(shù)據(jù)通信,串口無疑是最簡單的方式,今天我們使用 Qt 實現(xiàn)一個自己的串口調(diào)試助手。

 

實現(xiàn)目標

自己編寫一個基于 Qt 的串口調(diào)試軟件,可以實現(xiàn)本軟件與串口助手之間的通訊。

軟件發(fā)送的數(shù)據(jù),經(jīng)虛擬串口轉(zhuǎn)發(fā),能夠在串口助手中正確接收;

串口助手發(fā)送的數(shù)據(jù)可以在本軟件的接收文本框中顯示,進而實現(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]

 

本文源碼

后臺回復關(guān)鍵字“Qt-COM”,獲取本文涉及到的虛擬串口軟件及 Qt 工程源碼。

 

界面設計

利用 Qt Creator 新建一個 Project,模板選擇 Application--> Qt Widgets Application , 向?qū)е?Class Information 頁面中,Base class 選擇 QMainWindow 、 QWidget 、QDialog 都可以。

工程創(chuàng)建完畢,.ui 文件具體設計如下:

 

具體實現(xiàn)

導入串口通信模塊

從 Qt 5.1 版本開始,Qt 就有了自己的串口通訊類,之前版本需要使用第三方的串口通信類才行。

要想使用串口通信類,需要在 .pro 文件中添加 QT += serialport

 

顯示系統(tǒng)中所有串口號

顯示串口號列表的是一個 QComboBox 控件。

我們調(diào)用 QSerialPortInfo::availablePorts() 可以獲得一個 QList ,List 中的每一項 QSerialPortInfo 代表一個串口實例,該類中保存了系統(tǒng)中已有串口的端口名稱、系統(tǒng)位置、描述和供應商等信息。

遍歷系統(tǒng)中所有串口名的實現(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 變量中,這個變量的類型是 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()
// 設置要打開的串口名
void QSerialPort::setPortName(const QString &name)
// 設置串口通信的波特率
bool QSerialPort::setBaudRate(qint32 baudRate, QSerialPort::Directions directions = AllDirections)
// 設置串口通信的數(shù)據(jù)位,數(shù)據(jù)位一般為 8 位
bool QSerialPort::setDataBits(QSerialPort::DataBits dataBits)
// 設置串口通信的流控制,一般無需流控制
bool QSerialPort::setFlowControl(QSerialPort::FlowControl flowControl)
// 設置串口通信的奇偶校驗,一般選擇“無”
bool QSerialPort::setParity(QSerialPort::Parity parity)
// 設置串口通信的停止位,停止位一般為 1
bool QSerialPort::setStopBits(QSerialPort::StopBits stopBits)

在“打開串口”按鈕上右鍵彈出菜單中,選擇 轉(zhuǎn)到槽 ... ,在按鍵的 clicked() 事件中,添加串口打開的對應代碼。

串口通信類庫通信過程基本需要以下步驟,即:打開串口 --> 配置串口參數(shù)(波特率、數(shù)據(jù)位、停止位、奇偶校驗、流控等) --> 收發(fā)數(shù)據(jù)。

串口打開的具體實現(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);// 設置波特率和讀寫方向
        m_serialPort->setDataBits(QSerialPort::Data8);              // 數(shù)據(jù)位為 8 位
        m_serialPort->setFlowControl(QSerialPort::NoFlowControl);   // 無流控制
        m_serialPort->setParity(QSerialPort::NoParity);             // 無校驗位
        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)

這個函數(shù)是將以‘/0’結(jié)尾的字符串中的數(shù)據(jù)寫入設備(‘