Qt串口通信開(kāi)發(fā)之QSerialPort模塊Qt串口通信接收數(shù)據(jù)不完整的解決方法
在使用串口接收數(shù)據(jù)時(shí),當(dāng)數(shù)據(jù)量大的時(shí)候會(huì)出現(xiàn)數(shù)據(jù)接收不完整的情況。
因?yàn)榇跀?shù)據(jù)獲取函數(shù)readAll()由readyRead()信號(hào)觸發(fā),但readyRead()信號(hào)在串口讀到起始標(biāo)志時(shí)立即發(fā)送,并不保證一定是當(dāng)前所發(fā)數(shù)據(jù)的起始部分。
因此串口通信雙方在通信前應(yīng)制定好通信協(xié)議,規(guī)定好數(shù)據(jù)的起始和結(jié)束標(biāo)志,串口當(dāng)讀到完整的起始和結(jié)束標(biāo)志之后,才認(rèn)定讀完一條完整的數(shù)據(jù)。
本例中用串口定時(shí)發(fā)送當(dāng)前時(shí)間,用"#"表示數(shù)據(jù)的結(jié)尾,定時(shí)時(shí)間為0毫秒,即能發(fā)多快就發(fā)多快。
發(fā)送
void Widget::slotSendData() { QByteArray temp; temp.append(getCurrentTime()); temp.append("#"); serialPort->write(temp); }
接收
void Widget::slotReadData() { QByteArray temp = serialPort->readAll(); if(!temp.isEmpty()) { byteArray.append(temp); if(byteArray.contains("#")) { ui->textEditReceive->setText(byteArray.split('#').at(0)); byteArray = byteArray.right(byteArray.length()-byteArray.indexOf('#')-1); } } }
下面是一個(gè)通過(guò)串口傳輸圖片的例子
這里假設(shè)波特率為9600,那么一秒鐘就能傳輸9600/8=1200字節(jié)。代碼中將定時(shí)器設(shè)置為1秒,所以選擇的圖片應(yīng)該小于1200字節(jié)。
這里為了演示如何完整接收數(shù)據(jù),將圖片按照指定大小分段發(fā)送,在每段之后緊接著發(fā)送字符串“###”。
代碼如下所示:
void Widget::slotSendData() { matrix.rotate(90); QPixmap tempPixmap = pixmap.transformed(matrix); QBuffer buffer; tempPixmap.save(&buffer,"jpg"); ui->labelImage->setPixmap(tempPixmap); char *data=(char*)buffer.data().data(); int dataLength=buffer.data().length(); //打印圖片大小 qDebug()<<"Image Size:"<<dataLength; int standPacketSize=120; int packetSize=0; int packetNum=ceil(dataLength/120.0); if(dataLength>120) { for(int i=0;i<packetNum;i++) { if(standPacketSize*(i+1)<dataLength) { packetSize=standPacketSize; } else { packetSize=dataLength-standPacketSize*i; } serialPort->write(data,packetSize); data=data+packetSize; } } serialPort->write("###",3); }
運(yùn)行效果如下圖所示:
本文主要介紹了QSerialPort模塊Qt串口通信接收數(shù)據(jù)不完整的解決方法,更多關(guān)于Qt串口通信知識(shí)請(qǐng)查看下面的相關(guān)鏈接
相關(guān)文章
淺析C/C++ 中return *this和return this的區(qū)別
return *this返回的是當(dāng)前對(duì)象的克隆或者本身,return this返回當(dāng)前對(duì)象的地址,下面通過(guò)本文給大家介紹C/C++ 中return *this和return this的區(qū)別,感興趣的朋友一起看看吧2019-10-10C++ 創(chuàng)建桌面快捷方式 開(kāi)始菜單的實(shí)現(xiàn)代碼
這篇文章介紹了C++ 創(chuàng)建桌面快捷方式,開(kāi)始菜單的實(shí)現(xiàn)代碼,需要的朋友可以參考一下2013-06-06非常漂亮的新年祝福!C語(yǔ)言實(shí)現(xiàn)漂亮的煙花效果
非常漂亮的新年祝福!這篇文章主要介紹了C語(yǔ)言實(shí)現(xiàn)漂亮的煙花效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-02-02C++ Thread實(shí)現(xiàn)簡(jiǎn)單的socket多線(xiàn)程通信
本文主要介紹了C++ Thread實(shí)現(xiàn)簡(jiǎn)單的socket多線(xiàn)程通信,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-07-07C/C++實(shí)現(xiàn)crc碼計(jì)算和校驗(yàn)
循環(huán)冗余校驗(yàn)(Cyclic Redundancy Check, CRC)是一種根據(jù)網(wǎng)絡(luò)數(shù)據(jù)包或計(jì)算機(jī)文件等數(shù)據(jù)產(chǎn)生簡(jiǎn)短固定位數(shù)校驗(yàn)碼的一種信道編碼技術(shù)。本文主要介紹了C++實(shí)現(xiàn)crc碼計(jì)算和校驗(yàn)的方法,需要的可以參考一下2023-03-03C++中賦值運(yùn)算符與逗號(hào)運(yùn)算符的用法詳解
這篇文章主要介紹了C++中賦值運(yùn)算符與逗號(hào)運(yùn)算符的用法詳解,是C++入門(mén)學(xué)習(xí)中的基礎(chǔ)知識(shí),需要的朋友可以參考下2015-09-09