qt做串口调试助手——通信收发篇

qt实现串口调试助手

串口初始化

串口调试助手的初始化,一般包括波特率、端口号、奇偶校验、数据位和停止位。如果最终需要将串口商业化,一般这些参数都会交给用户自定义,但我们写了自己用,且上位机和下位机同时开发,则除了端口号,其余参数都会在代码里面定义好。
端口号一般采用自动扫描更新可用串口的方式,比直接列举COM1、COM2等要好很多。

	//参数初始化
 	global_port.setBaudRate(QSerialPort::Baud115200);  //波特率
    global_port.setParity(QSerialPort::NoParity)  ;   //无奇偶校验
    global_port.setDataBits(QSerialPort::Data8);    //8位数据位
    global_port.setStopBits(QSerialPort::OneStop);  // 1位停止位
    //定时扫描和更新串口
    timer = new QTimer;
    connect(timer,&QTimer::timeout,this,&MainWindow::showValidPort);//实时更新端口号
    timer->start(1000); //每1000ms定时检测串口状态
    foreach(const QSerialPortInfo &info, QSerialPortInfo::availablePorts()) //扫描可用串口
    portStringList += info.portName();
    ui->comboVolt1->addItems(portStringList);//将扫描的端口号添加到下拉框

自定义通信协议

为了避免数据传输过程中丢失,我们不会直接接收并使用下位机发送的数据,而是拟定通信协议,包括设置帧头、数据位长度、数据、校验位、帧尾等。这样可以通过对帧头、帧尾的检验,防止数据在传输过程中有丢失。
下图是我根据自己的需求拟定的通信协议,一般如果能通过校验位的校验和帧头帧尾的检测,就说明数据在传输过程中没有丢失。
qt做串口调试助手——通信收发篇_第1张图片

上位机的接收

通信协议拟定好了,上位机的接收也是有讲究的,因为上位机不可能一次性正好接收以上的全部内容,而且我们传输的数据也有可能包含帧头帧尾等对应的ASCII码,为了防止误接收,一般我们设置一个缓冲区去接收数据,接收完一帧之后在处理
这下面的代码其实还有点问题,就是出现校验失败等问题之后,没有清除缓冲区重新接收,后面会再改进

 if((!array.contains("{"))&&buffer.isNull())//异常:头部丢失
        {
            return ;
        }
        //第一种,有头无尾,附加
        if(array.contains("{")&&(!array.contains("}")))
        {
           // buffer.clear();
            buffer.append(array);
        }
        //第二种:无头无尾且变量已有内容,数据中段部分,继续附加即可
        if((!array.contains("{"))&&(!array.contains("}"))&&(!buffer.isNull()))
        {
            buffer.append(array);
        }
        //第三种:无头有尾且变量已有内容,已完整读取,附加后可做处理
        if((!array.contains("{"))&&(array.contains("}"))&&(!buffer.isNull()))
        {
             buffer.append(array);
             is_readall = true;
        }
        //第四种:有头有尾(一段完整的内容),已完整读取
        if((array.contains("{"))&&(array.contains("}")))
        {
             buffer.append(array);
             is_readall = true;
        }
        if(is_readall)//缓存区有头有尾,已完整读取
        {
            qDebug() <<"buffer_size:";
            qDebug() <= BUFFERSIZE)  array_signal.at(i).pop_back();
                uint buf_0 = ((uint)buffer.at(4 * i+start)<< 24)& 0xff000000;
                uint buf_1 = ((uint)(buffer.at(4 * i + 1+start)) << 16)& 0x00ff0000;
                uint buf_2 = ((uint)(buffer.at(4 * i + 2+start)) << 8)  & 0x0000ff00;
                uint buf_3 = ((uint)(buffer.at(4 * i + 3+start)) )&0x000000ff ;
                uint buf = buf_0 + buf_1 + buf_2 + buf_3;
                sum+=buf_3;//校验和

                orginal_signal.at(i).insert(orginal_signal.at(i).begin(),
                    (float)(buf));//

                    buf = buf/1000;
                array_signal.at(i).insert(array_signal.at(i).begin(),
                    (float)(buf));//
                if(max_range[i]buf) min_range[i] = buf;
            }

            is_readall = false;
            sum = sum&0xff;
            checksum = (uint)(buffer.at(buffer.size()-2))&0xff;//检查校验位
            if(checksum == sum)
            {
                qDebug() <<"checkcum successful!";
            }
            else
            {
                qDebug() <<"checkcum failed!";
            }
            if((uint)(buffer.at(buffer.size()-1))== 125)//帧尾
            {
                qDebug() <<"end!";
            }
            buffer.clear();

下位机的发送

串口助手肯定是需要下位机通过串口将数据发给上位机,不管是上位机的发送还是下位机的接收,都要跟通信协议对应上,相比之下,下位机的发送就很简单了,如果想要定时发送,就开个定时器,否则直接while(1)里面发送就行了。
以下是对应的下位机发送格式,使用的是STM32F1的板子

void send_resistance_value(u32 *value,int len)
{
	u8 temp1;
	u32 temp,sum = 0;
	int i = 0;
	USART_SendData(USART1, '{');//帧头1
	while(USART_GetFlagStatus(USART1,USART_FLAG_TC)!=SET);//等待发送结束
	
	USART_SendData(USART1, len&0xff);//帧头1
	while(USART_GetFlagStatus(USART1,USART_FLAG_TC)!=SET);//等待发送结束
	for(;i>24;
		USART_SendData(USART1, temp1);//向串口1发送最高位
		while(USART_GetFlagStatus(USART1,USART_FLAG_TC)!=SET);//等待发送结束
		
		temp1 = (temp&0xff0000)>>16;
		USART_SendData(USART1, temp1);//向串口1发送次高位
		while(USART_GetFlagStatus(USART1,USART_FLAG_TC)!=SET);//等待发送结束
		
		temp1 = (temp&0xff00)>>8;
		USART_SendData(USART1, temp1);//向串口1发送次低位
		while(USART_GetFlagStatus(USART1,USART_FLAG_TC)!=SET);//等待发送结束
		
		temp1 = (temp&0xff);
		USART_SendData(USART1, temp1);//向串口1发送最低位
		while(USART_GetFlagStatus(USART1,USART_FLAG_TC)!=SET);//等待发送结束
		
		sum+=temp1;//校验和		
	}
	
	temp1 = sum&0xff;
	USART_SendData(USART1, temp1);//发送校验和
	while(USART_GetFlagStatus(USART1,USART_FLAG_TC)!=SET);//等待发送结束
	
	USART_SendData(USART1, '}');//帧尾
	while(USART_GetFlagStatus(USART1,USART_FLAG_TC)!=SET);//等待发送结束
}

你可能感兴趣的:(qt,qt,单片机,开发语言)