实验2 实时显示单片机的参数(整数、小数、中文 自定义协议方式)

1. 实验目的

单片机按照自定义协议格式发送控制命令,串口屏实时显示单片机的参数,包括整数、小数、和中文字符。实现效果和实验一相同。

2. 页面设计

(1)进度条控件,以进度条的方式显示出SOC值

(2)整数控件,显示具体SOC值

(3)文本控件,显示电量状态

(4)浮点数控件,显示电压值

(5)整数控件,显示电流值

(6)协议解析器控件,编写脚本代码,解析自定义的协议

选中控件,可以在右侧属性栏查看控件的各个属性。详细属性说明可参考 第四章 控件的介绍和用法。

资源区:

一共使用4张图片,home作为页背景图片,另外三张图片为进度条不同状态的颜色。

字体字体除了工程自带的字库Tahoma_4x_ASCII(包括英文和数字),另外创建了一个中文的字库,双击这个字库,可以看到详细的信息。字库详细制作方法可以参考第三章的添加字库小节。

3. 串口屏协议处理

该例程使用了协议解析器控件 。串口屏每次收到一帧数据,都会触发一次协议解析器的脚本,和单片机的空闲中断函数类似。

协议定义(仅仅是示例,仅供参考)

帧头(2字节)

命令码(1字节)

数据(2字节)

校验(2字节)

备注

A5 43

1

2byte

2byte

设置电压 使用时除100,如3.14V,传输时是314

A5 43

2

2byte

2byte

设置电流

A5 43

3

2byte

2byte

设置SOC

数据和校验 使用小端模式,即低字节在前,高字节在后

校验方式为CRC16,多项式8005

选中控件,在事件编辑器串口可以看到数据的解析过程,代码如下:


int crc_val;
int crc_data;
float V;

if((protocol7.rxBuf[0] == 0xA5)&&(protocol7.rxBuf[1] == 0x43))  //①判断帧头
{
    if(protocol7.rxLen == 7)  //②判断帧长度
    {
        crc_data = (protocol7.rxBuf[6]<<8) | protocol7.rxBuf[5];
        crc_val  = crc16(protocol7.rxBuf,0,5);//③
        if(crc_data == crc_val) //CRC判断
        {
            if(protocol7.rxBuf[2] == 0x01)  //电压
            {
                V = (protocol7.rxBuf[4]<<8) | protocol7.rxBuf[3];
                numVf.valf = V/100;                
            }
            else if(protocol7.rxBuf[2] == 0x02)  //电流
            {
                numA.val = (protocol7.rxBuf[4]<<8) | protocol7.rxBuf[3];
            }
            else if(protocol7.rxBuf[2] == 0x03)  //SOC
            {
                psoc.val = (protocol7.rxBuf[4]<<8) | protocol7.rxBuf[3];
                nsoc.val = psoc.val;
                
                if(nsoc.val > 80)
                {
                    psoc.fgImg = user.image.blue; 
                    text.txt = "电量充足";
                }
                else if(nsoc.val > 20)
                {
                    psoc.fgImg = user.image.yellow;
                    text.txt = "电量正常";
                }
                else
                {
                    psoc.fgImg = user.image.red;
                    text.txt = "电量过低";
                }
            }            
        }
    }
}

①接收到串口数据缓存在协议解析器控件的rxBuf里,是一个数组,数组长度是协议解析器的一个属性,可配置。决定接收一帧数据的最大长度,最大可设置512字节。

② 属性rxLen是本次接收到数据长度。

③ crc16是一个内部函数,用于计算CRC校验值。具体使用方法 请参考 第五章 函数的介绍和用法。具体项目中,可选择使用或者不使用校验。

④ psoc.fgImg = user.image.blue;这里是修改进度条的图片,也可以写成这样psoc.fgImg = 2;效果是一样的。

4. 下载验证

编译成功后,点击下载按钮,选择正确的端口号和波特率,下载到串口屏。可以看到显示如下显示:

测试时,可以先通过电脑串口助手给屏发指令来调试。

注意:发送时勾选hex发送

测试的命令

A5 43 01 7C 01 91 86 //设置电压值3.80V
A5 43 02 0F 00 AE 2C //设置电流值15mA
A5 43 03 0F 00 B9 AC //设置SOC值15%

串口屏显示如下,表示串口屏收到数据并正确处理。

实际应用时,是MCU和串口屏通信。本实验使用的单片机型号是STM32F103TB,改例程仅作供参考,开发环境keil V5.24。其他MCU的串口发送方式也类似。

部分代码

int main(void)
{
	u16 crc_val;
	
	
	Sys_SetRcc();                                 //设置系统使用内部时钟
	delay_init(64);	    	                      //延时函数初始化	 
	usart1_init();                                //初始化串口 波特率115200
	
	while(1)
	{
		//设置电压值
		str[0] = 0xA5;
		str[1] = 0x43;
		str[2] = 0x01;
		str[3] = (V>>0) & 0xFF;
		str[4] = (V>>8) & 0xFF;
		crc_val = Get_CRC16(5,str);
		str[5] = (crc_val>>0) & 0xFF;
		str[6] = (crc_val>>8) & 0xFF;		
		
		USART_OUT(USART1, (uint8_t*)str,7);	
		delay_ms(1); //命令之间需要加上小延时分开
		
		
		
		//设置电流值
		str[0] = 0xA5;
		str[1] = 0x43;
		str[2] = 0x02;
		str[3] = (I>>0) & 0xFF;
		str[4] = (I>>8) & 0xFF;
		crc_val = Get_CRC16(5,str);
		str[5] = (crc_val>>0) & 0xFF;
		str[6] = (crc_val>>8) & 0xFF;		
		
		
		USART_OUT(USART1, (uint8_t*)str,7);	
		delay_ms(1); //命令之间需要加上小延时分开

		//设置SOC
		str[0] = 0xA5;
		str[1] = 0x43;
		str[2] = 0x03;
		str[3] = (SOC>>0) & 0xFF;
		str[4] = (SOC>>8) & 0xFF;
		crc_val = Get_CRC16(5,str);
		str[5] = (crc_val>>0) & 0xFF;
		str[6] = (crc_val>>8) & 0xFF;	
		
		USART_OUT(USART1, (uint8_t*)str,7);	
		delay_ms(1); //命令之间需要加上小延时分开		
		
		//模拟数据变化
		V+=8;
		if(V > 500)V = 300; //电压值范围3~5V
		if(++I > 100)I = 0;
		if(++SOC > 100)SOC = 0;

		delay_ms(200);
			
	}
	
}

单片机部分代码很简单,按照协议格式组帧发送即可。注意每条命令之间需要加一点点延时用于串口屏区分。

总结:

实验二和实验一,最后的显示结果都是一样的。系统指令的方式适合界面不复杂的情况,无需设计通信协议,按照串口屏内置的系统指令格式来发送即可;自定义协议方式体现为更加灵活,C语言编写脚本,通信效率更高,一帧数据可给多个控件赋值,适用于大多数项目。推荐使用自定义协议方式。两种方式混合使用也可以。