STM32
直播中

康大争

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

求大佬分享USART串口接收程序

求大佬分享USART串口接收程序

回帖(1)

李超

2021-12-9 11:07:06
STM32串口接收的程序和8051单片机一样,有2种接收方式:中断处理和查询方式。
试过两种方式之后发现,中断处理方式更方便,可以扩展多个数据的接收。推荐中断方式。
  我已经将本资料内容整理后录制了视频《STM32入门100步》
其中几集涉及到USART的原理和使用:(B站在线)
  第33步视频——33:USART驱动程序:https://www.bilibili.com/video/av18222153/?p=35
第34步视频——34:USART发送程序:https://www.bilibili.com/video/av18222153/?p=36
第35步视频——35:USART接收程序:https://www.bilibili.com/video/av18222153/?p=37
第36步视频——36:USART控制程序:https://www.bilibili.com/video/av18222153/?p=38
  全部视频下载:http://www.doyoung.net/YT/index.html
  相关程序下载:http://www.doyoung.net/YT/index.html 编号YC008
    查询方式的接收:

  1,在usart.c文件里,uart_init()函数中
  
void uart_init(u32 bound){
    //GPIO端口设置
    GPIO_InitTypeDef GPIO_InitStructure;
    USART_InitTypeDef USART_InitStructure;
    NVIC_InitTypeDef NVIC_InitStructure;
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1|RCC_APB2Periph_GPIOA, ENABLE);    //使能USART1,GPIOA时钟
     //USART1_TX   PA.9
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; //PA.9
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;    //复用推挽输出
    GPIO_Init(GPIOA, &GPIO_InitStructure);
    //USART1_RX      PA.10
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//浮空输入
    GPIO_Init(GPIOA, &GPIO_InitStructure);
   //Usart1 NVIC 配置
    NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ;//抢占优先级3
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;        //子优先级3
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;            //IRQ通道使能
    NVIC_Init(&NVIC_InitStructure);    //根据指定的参数初始化VIC寄存器
   //USART 初始化设置
    USART_InitStructure.USART_BaudRate = bound;//一般设置为9600;
    USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字长为8位数据格式
    USART_InitStructure.USART_StopBits = USART_StopBits_1;//一个停止位
    USART_InitStructure.USART_Parity = USART_Parity_No;//无奇偶校验位
    USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//无硬件数据流控制
    USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;    //收发模式
    USART_Init(USART1, &USART_InitStructure); //初始化串口
    USART_ITConfig(USART1, USART_IT_RXNE, DISABLE);//开启ENABLE/关闭DISABLE中断
    USART_Cmd(USART1, ENABLE);                    //使能串口
}
在串口中断设置一项中,要关闭中断:
USART_ITConfig(USART1, USART_IT_RXNE, DISABLE);//开启ENABLE/关闭DISABLE中断  

  
2,把USART1_IRQHandler()函数里的所有内容屏蔽或者删除。(这一步是为了让程序简洁,防止误会,不删除也不影响)
void USART1_IRQHandler(void){ //串口1中断服务程序(固定的函数名不能修改)//    u8 Res;//    if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET){  //接收中断//        Res =USART_ReceiveData(USART1);//(USART1->DR);    //读取接收到的数据//        printf("收到数据 %c rn",Res); //把收到的数据以 a符号变量 发送回电脑//    }}12345673,在主程序开头加入串口初始化函数并给出波特率值。在循环程序部分加入查询程序。
int main (void){//主程序    u8 Res;    RCC_Configuration();    delay_init(); //延时函数初始化    uart_init(115200); //串口初始化,参数中写波特率    while(1){        if(USART_GetFlagStatus(USART1,USART_FLAG_RXNE) != RESET){  //查询串口待处理标志位            Res =USART_ReceiveData(USART1);//(USART1->DR);    //读取接收到的数据            printf("收到数据 %c rn",Res); //把收到的数据以 a符号变量 发送回电脑        }    }}123456789101112注意:在使用查询方式时,使用的是USART_GetFlagStatus()函数
if(USART_GetFlagStatus(USART1,USART_FLAG_RXNE) != RESET){
//查询串口待处理标志位

要与接收中断标志位判断USART_GetITStatus()函数区别开。
if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET){ //接收中断标志位
4,在电脑的超级终端中键入“1、4、8、y”等字符,在超级终端上会显示如下图。


  

  

  

   
中断方式的接收:

1,在usart.c文件里,uart_init()函数中

void uart_init(u32 bound){
    //GPIO端口设置
    GPIO_InitTypeDef GPIO_InitStructure;
    USART_InitTypeDef USART_InitStructure;
    NVIC_InitTypeDef NVIC_InitStructure;
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1|RCC_APB2Periph_GPIOA, ENABLE);    //使能USART1,GPIOA时钟
     //USART1_TX   PA.9
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; //PA.9
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;    //复用推挽输出
    GPIO_Init(GPIOA, &GPIO_InitStructure);
    //USART1_RX      PA.10
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//浮空输入
    GPIO_Init(GPIOA, &GPIO_InitStructure);
   //Usart1 NVIC 配置
    NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ;//抢占优先级3
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;        //子优先级3
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;            //IRQ通道使能
    NVIC_Init(&NVIC_InitStructure);    //根据指定的参数初始化VIC寄存器
   //USART 初始化设置
    USART_InitStructure.USART_BaudRate = bound;//一般设置为9600;
    USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字长为8位数据格式
    USART_InitStructure.USART_StopBits = USART_StopBits_1;//一个停止位
    USART_InitStructure.USART_Parity = USART_Parity_No;//无奇偶校验位
    USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//无硬件数据流控制
    USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;    //收发模式
    USART_Init(USART1, &USART_InitStructure); //初始化串口
    USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);//开启ENABLE/关闭DISABLE中断
    USART_Cmd(USART1, ENABLE);                    //使能串口
}
2,在串口中断处理函数中,定义一个u8的8位字节变量,用于存放串口发来的数据。
然后再把收到的数据发回电脑(这个步骤是用于验证,不发回也可以)。

void USART1_IRQHandler(void){ //串口1中断服务程序(固定的函数名不能修改)
    u8 Res;
    if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET){  //接收中断
        Res =USART_ReceiveData(USART1); //(USART1->DR);    //读取接收到的数据
        printf("收到数据 %c rn",Res); //把收到的数据以 a符号变量 发送回电脑
    }
}1234567注意:这里使用 %c 的a符号变量,意思是 单个字符。
3,在main.c里的main函数中,只要在开头加入串口初始化函数并给出波特率值,在循环程序中不需要写任何程序。
int main (void){//主程序
    RCC_Configuration();
    delay_init(); //延时函数初始化
    uart_init(115200); //串口初始化,参数中写波特率
    while(1){

    //在USART1中断处理程序USART1_IRQHandler()中完成

    }
}
123456789104,在电脑的超级终端中键入“1、4、8、y”等字符,在超级终端上会显示如下图。


  

  

  

  

    关于USART串口的使用还有很多内容要写,请找本博客的其他文章,应该会有更新。
举报

更多回帖

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