一、串口介紹
1.波特率
衡量通信速度的參數,表示每秒鍾傳送的bit的個數。例如9600波特表示每秒鍾發送9600個bit。常見的波特率:9600(QSerialPort::Baud9600),19200(QSerialPort::Baud19200),38400(QSerialPort::Baud38400),115200(QSerialPort::Baud115200)
2.數據位
衡量通信中實際數據位的參數,當計算機發送一個信息包,實際包含的有效數據位個數。常見設置:5(QSerialPort::Data5),6(QSerialPort::Data6),7(QSerialPort::Data7),8(QSerialPort::Data8)
3.停止位
用於表示單個包的最后一位。典型的值為1和2位。常見設置:1(QSerialPort::OneStop),1.5(QSerialPort::OneAndHalfStop),2(QSerialPort::TwoStop)
4.奇偶校驗位
串口通信中一種檢錯方式。常用的檢錯方式有:偶、奇校驗。常見設置:None(QSerialPort::NoParity),Even(QSerialPort::EvenParity),Odd(QSerialPort::OddParity)等
5.流控
QT上還提供設置數據流控
注:串口數據流控
轉載:https://blog.csdn.net/lijia11080117/article/details/51371893
1.流控制在串行通訊中的作用
這里講到的“流”,當然指的是數據流。數據在兩個串口之間傳輸時,常常會出現丟失數據的現象,或者兩台計算機的處理速度不同,如台式機與單片機之間的通訊,接收端數據緩沖區已滿,則此時繼續發送來的數據就會丟失。現在我們在網絡上通過modem進行數據傳輸,這個問題就尤為突出。流控制能解決這個問題,當接收端數據處理不過來時,就發出“不再接收”的信號,發送端就停止發送,直到收到“可以繼續發送”的信號再發送數據。因此流控制可以控制數據傳輸的進程,防止數據的丟失。 pc機中常用的兩種流控制是硬件流控制(包括rts/cts、dtr/cts等)和軟件流控制xon/xoff(繼續/停止),下面分別說明。
2.硬件流控制
硬件流控制常用的有rts/cts流控制和dtr/dsr(數據終端就緒/數據設置就緒)流控制。
硬件流控制必須將相應的電纜線連上,用rts/cts(請求發送/清除發送)流控制時,應將通訊兩端的rts、cts線對應相連,數據終端設備(如計算機)使用rts來起始調制解調器或其它數據通訊設備的數據流,而數據通訊設備(如調制解調器)則用cts來起動和暫停來自計算機的數據流。
這種硬件握手方式的過程為:我們在編程時根據接收端緩沖區大小設置一個高位標志(可為緩沖區大小的75%)和一個低位標志(可為緩沖區大小的25%),當緩沖區內數據量達到高位時,我們在接收端將cts線置低電平(送邏輯0),當發送端的程序檢測到cts為低后,就停止發送數據,直到接收端緩沖區的數據量低於低位而將cts置高電平。rts則用來標明接收設備有沒有准備好接收數據。常用的流控制還有還有dtr/dsr(數據終端就緒/數據設置就緒)。
3.軟件流控制
由於電纜線的限制,我們在普通的控制通訊中一般不用硬件流控制,而用軟件流控制。一般通過xon/xoff來實現軟件流控制。常用方法是:當接收端的輸入緩沖區內數據量超過設定的高位時,就向數據發送端發出xoff字符(十進制的19或control-s,設備編程說明書應該有詳細闡述),發送端收到 xoff字符后就立即停止發送數據;當接收端的輸入緩沖區內數據量低於設定的低位時,就向數據發送端發出xon字符(十進制的17或control- q),發送端收到xon字符后就立即開始發送數據。
代碼展示
//初始化波特率
ui->Net_S_BaudrateCombo->addItem(tr("自定義"));
ui->Net_S_BaudrateCombo->addItem(QStringLiteral("9600"), QSerialPort::Baud9600);
ui->Net_S_BaudrateCombo->addItem(QStringLiteral("19200"), QSerialPort::Baud19200);
ui->Net_S_BaudrateCombo->addItem(QStringLiteral("38400"), QSerialPort::Baud38400);
ui->Net_S_BaudrateCombo->addItem(QStringLiteral("115200"), QSerialPort::Baud115200);
ui->Net_S_BaudrateCombo->setCurrentText(tr("38400"));
//初始化數據位
ui->Net_S_DataCombo->addItem(QStringLiteral("5"), QSerialPort::Data5);
ui->Net_S_DataCombo->addItem(QStringLiteral("6"), QSerialPort::Data6);
ui->Net_S_DataCombo->addItem(QStringLiteral("7"), QSerialPort::Data7);
ui->Net_S_DataCombo->addItem(QStringLiteral("8"), QSerialPort::Data8);
ui->Net_S_DataCombo->setCurrentText(tr("8"));
//初始化停止位
ui->Net_S_StopCombo->addItem(QStringLiteral("1"), QSerialPort::OneStop);
ui->Net_S_StopCombo->addItem(tr("1.5"), QSerialPort::OneAndHalfStop);
ui->Net_S_StopCombo->addItem(QStringLiteral("2"), QSerialPort::TwoStop);
ui->Net_S_StopCombo->setCurrentText(tr("1"));
//初始化校驗位
ui->Net_S_ChkCombo->addItem(tr("None"), QSerialPort::NoParity);
ui->Net_S_ChkCombo->addItem(tr("Even"), QSerialPort::EvenParity);
ui->Net_S_ChkCombo->addItem(tr("Odd"), QSerialPort::OddParity);
ui->Net_S_ChkCombo->addItem(tr("Mark"), QSerialPort::MarkParity);
ui->Net_S_ChkCombo->addItem(tr("Space"), QSerialPort::SpaceParity);
ui->Net_S_ChkCombo->setCurrentText(tr("None"));
//初始化流控
ui->Net_S_FlowCombo->addItem(tr("None"), QSerialPort::NoFlowControl);
ui->Net_S_FlowCombo->addItem(tr("RTS/CTS"), QSerialPort::HardwareControl);
ui->Net_S_FlowCombo->addItem(tr("XON/XOFF"), QSerialPort::SoftwareControl);
ui->Net_S_FlowCombo->setCurrentText(tr("None"));
//設置波特率和讀寫方向
qint32 baudRate;
if (ui->Net_S_BaudrateCombo->currentIndex() == 4){
baudRate = ui->Net_S_BaudrateCombo->currentText().toInt();
}else{
baudRate = static_cast<QSerialPort::BaudRate>(ui->Net_S_BaudrateCombo->itemData(ui->Net_S_BaudrateCombo->currentIndex()).toInt());
}
m_serialPort->setBaudRate(baudRate,QSerialPort::AllDirections);
//數據位
QSerialPort::DataBits dataBits = static_cast<QSerialPort::DataBits>(ui->Net_S_DataCombo->itemData(ui->Net_S_DataCombo->currentIndex()).toInt());
m_serialPort->setDataBits(dataBits);
//停止位
QSerialPort::StopBits stopBits = static_cast<QSerialPort::StopBits>(ui->Net_S_StopCombo->itemData(ui->Net_S_StopCombo->currentIndex()).toInt());
m_serialPort->setStopBits(stopBits);
//校驗位
QSerialPort::Parity parity = static_cast<QSerialPort::Parity>(ui->Net_S_ChkCombo->itemData(ui->Net_S_ChkCombo->currentIndex()).toInt());
m_serialPort->setParity(parity);
//流控制
QSerialPort::FlowControl flowControl = static_cast<QSerialPort::FlowControl>(ui->Net_S_FlowCombo->itemData(ui->Net_S_FlowCombo->currentIndex()).toInt());
m_serialPort->setFlowControl(flowControl);
初始化串口
1、從本機上查詢所有串口,並在界面顯示
QStringList list;
QList<QSerialPortInfo> serialPortInfoList = QSerialPortInfo::availablePorts();//讀取所有可用的串口信息
int intID = 0;
foreach(QSerialPortInfo serialPortInfo,serialPortInfoList) //打印出端口信息
{
QString strComboShow = (serialPortInfo.portName() + ":" + serialPortInfo.description());
ui->Net_S_COMCombo->insertItem(intID,strComboShow,serialPortInfo.portName());
}
QSerialPortInfo 類中可獲取串口端口的名稱,如:COM1,COM2等。還有相關串口描述。
2、打開串口
m_serialPort->setPortName("串口名字");//當前選擇的串口名字
if(!m_serialPort->open(QIODevice::ReadWrite))//用ReadWrite 的模式嘗試打開串口
{
QMessageBox::warning(this, "警告", "打開串口失敗");
return;
}
3、串口接收數據
// 串口數據到來時,會觸發QSerialPort::readyRead事件,添加相應的響應函數
QObject::connect(m_serialPort, &QSerialPort::readyRead, this, &Nahai::serialReceiveData);
void serialReceiveData()
{
QByteArray buffer = m_serialPort->readAll();
//處理你要處理的數據
}
4、串口發送數據
if(m_serialPort->isOpen())
{
m_serialPort->write(baPacket,baPacket.size());//QByteArray baPacket:為你要發送字節數組
}