STM32
直播中

陈秀英

7年用户 1222经验值
私信 关注
[问答]

如何使用python中的struct.pack函数打包数据并通过串口发送呢

如何使用python中的struct.pack函数打包数据并通过串口发送呢?
big-Endian和little-Endian的区别在哪?

回帖(1)

王健

2021-12-6 15:30:58
Python串口数据打包发送STM32接收数据解析

尝试使用python中的struct.pack函数打包数据通过串口发送,由STM32接收解析。
1、 struct.pack:
struct.pack用于将Python的值根据格式符,转换为字符串(因为Python中没有字节(Byte)类型,可以把这里的字符串理解为字节流,或字节数组)。其函数原型为:struct.pack(fmt, v1, v2, …),参数fmt是格式字符串,关于格式字符串的相关信息在下面有所介绍。v1, v2, …表示要转换的python值。
2、 struct.unpack:
struct.unpack做的工作刚好与struct.pack相反,用于将字节流转换成python数据类型。它的函数原型为:struct.unpack(fmt, string),该函数返回一个元组。
上述fmt中,支持的格式为:





注1.q和Q只在机器支持64位操作时有意思;
注2.每个格式前可以有一个数字,表示个数;
注3.s格式表示一定长度的字符串,4s表示长度为4的字符串,但是p表示的是pascal字符串;
注4.P用来转换一个指针,其长度和机器字长相关;
注5.最后一个可以用来表示指针类型的,占4个字节;
为了同c中的结构体交换数据,还要考虑有的c或c++编译器使用了字节对齐,通常是以4个字节为单位的32位系统,故而struct根据本地机器字节顺序转换.可以用格式中的第一个字符来改变对齐方式.定义如下:





python端代码如下:

import serial
import struct
import time


# 串口号 默认为 /dev/ttyUSB0
#ser_dev = '/dev/ttyUSB1'


ser_dev ='com1'
# 创建一个串口实例
ser = serial.Serial(ser_dev, 115200, timeout=1, bytesize=8)


def pack_bin_data(bottom_degree, top_degree):
    '''
    h: unsigned short bit=2
    b: unsigned char (byte): bit =1
    '''
    bin_data = struct.pack(">iiBB", # 大端存储
                      int(a),       # 数据1
                      int(b),       # 数据2
                      0x0D,         # 帧尾
                      0x0A)         # 结束符 'n = 0x0A
    return bin_data


ser.write(bin_data )
big-Endian和little-Endian区别:
小端:较高的有效字节存放在较高的的存储器地址,较低的有效字节存放在较低的存储器地址。
大端:较高的有效字节存放在较低的存储器地址,较低的有效字节存放在较高的存储器地址。
例如0x12345678 ,在大端模式的排列:0x01(低地址),0x23,0x45,0x67,0x89(高地址)。
在小端模式的排列:0x89(低地址),0x67,0x45,0x23,0x01(高地址)。
32端解析函数如下:
/
//串口1中断服务程序
void USART1_IRQHandler(void)            
{
        u8 USART_RX_BUF[USART_REC_LEN];   //接收缓冲,最大USART_REC_LEN个字节.
        u16 USART_RX_STA=0;   //接收状态标记          
        u8 Res;
#if SYSTEM_SUPPORT_OS         //如果SYSTEM_SUPPORT_OS为真,则需要支持OS.
        OSIntEnter();   
#endif
       
        if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)  //接收中断(接收到的数据必须是0x0d 0x0a结尾)
                {
                Res =USART_ReceiveData(USART1);        //读取接收到的数据
               
                if((USART_RX_STA&0x8000)==0)//接收未完成
                {
                        if(USART_RX_STA&0x4000)//接收到了0x0d
                        {
                                if(Res!=0x0a)
                                        USART_RX_STA=0;//接收错误,重新开始
                                else
                                        USART_RX_STA|=0x8000;//接收完成了
                        }
                        else //还没收到0X0D
                        {       
                                if(Res==0x0d)USART_RX_STA|=0x4000;
                                else
                                {
                                        USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;
                                        USART_RX_STA++;
                                        if(USART_RX_STA>(USART_REC_LEN-1))
                                                USART_RX_STA=0;//接收数据错误,重新开始接收          
                                }                 
                        }
                }                    
     }
               
#if SYSTEM_SUPPORT_OS         //如果SYSTEM_SUPPORT_OS为真,则需要支持OS.
        OSIntExit();                                                                                           
#endif
         
}




void Data_depack(u8 *data_buf,u8 num)
{
        int btm_degree,top_degree;
    /* 读取btm角度原始数据  */
    btm_degree = (int)(*(data_buf+0)) | (int)(*(data_buf+1)) | (int)(*(data_buf+2)) | (int)(*(data_buf+3))
   
    /* 读取top角度原始数据 */
    top_degree = (int)(*(data_buf+4)) | (int)(*(data_buf+5)) | (int)(*(data_buf+6)) | (int)(*(data_buf+7)) ;
}
举报

更多回帖

发帖
×
20
完善资料,
赚取积分