STM32之 串口

news2025/1/23 9:22:17

串口通信

串行接口简称串口,也称串行通信接口或串行通讯接口(通常指COM接口),是采用串行通信方 式的扩展接口。串行接口(Serial Interface)是指数据一位一位地顺序传送。其特点是通信线路简 单,只要一对传输线就可以实现双向通信(可以直接利用电话线作为传输线),从而大大降低了成本,特别适用于远距离通信,但传送速度较慢。

  • 设备之间的一种通信的方式(数据交互)

  • 数据只能一位一位的进行传输

  • 全双工通信

  • 数据传输速度较慢

全双工通信(打电话)

通信允许数据在两个方向上同时传输,它在能力上相当于两个单工通信方式的结合。全双工指可以同时(瞬时)进行信号的双向传输(A→B且B→A)。指A→B的同时B→A,是瞬时同步的。通信允许数据在两个方向上同时传输,它在能力上相当于两个单工通信方式的结合。全双工指可以同时(瞬时)进行信号的双向传输(A→B且B→A)。指A→B的同时B→A,是瞬时同步的。

TTL电平

TTL是Transistor-Transistor Logic,即晶体管-晶体管逻辑的简称,它是计算机处理器控制的设备 内部各部分之间通信的标准技术。TTL电平信号应用广泛,是因为其数据表示采用二进制规定, +5V等价于逻辑”1”,0V等价于逻辑”0”。

字符帧

字符帧也叫数据帧,一帧数据由起始位、数据位、奇偶校验位和停止位组成,对于异步通信,要能够进行数据的正确传送,通信双方必须先约定好传送数据的速率和传送数据的组织格式,即波特率和字符帧的格式。

  • 起始位:每开始一次通信时发送方先发出一个逻辑”0”的信号(低电平),表示传输字符的开始。

  • 数据位:数据位可以是5、6、7、8,9位等,构成一个字符(一般都是8位)。如ASCII码(7位),扩展BCD码(8位)。先发送最低位,最后发送最高位,使用低电平表示‘0’高电平表示‘1’完成数据位的传输。

  • 奇偶校验位:计算数据位中“1”的位数应为偶数(偶校验)或奇数(奇校验),以此来校验数据传送的正确性,校验方式:

  1. 无校验 (no parity)。

  1. 奇校验 (odd parity): 如果数据位中“1”的数目是偶数,则校验位为“1”,如果“1的数目是奇数,校验位为“0”。

  1. 偶校验 (even parity): 如果数据为中“1”的数目是偶数,则校验位为“0”,如果为奇数,校验位为“1”。

  1. mark parity: 校验位始终为1 (不常用)。

  1. parity: 校验位始终为0 (不常用)。

  • 停止位:一个字符数据的结束标志。可以是1位、1.5位、2位的高电平。

总之,使用UART串口协议传输数据时,需要规定双方的传输速率(波特率)一致,数据格式(起始位,数据位,奇偶校验为、停止位)也要保持一致。例如传递字符a时,字符a的一帧信息由起始位、数据位(字符a)、奇偶校验位(可选)、停止位组成。

串口之非中断法

STM32的hal库关于串口的函数

HAL_UART_Transmit()

以阻塞的方式向串口发送数据,设定指定时间,超过指定时间后不再发送此次数据,不产生中断

原型:HAL_StatusTypeDef HAL_UART_Transmit(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size, uint32_t Timeout)

参数:
UART_HandleTypeDef *huart:串口句柄
uint8_t *pData:向串口发送的数据的地址
uint16_t Size:向串口发送的数据的大小
uint32_t Timeout:此次数据发送的时间

实例:HAL_UART_Transmit(&huart1,(uint8_t *)"jiangxiaoya\n",strlen("jiangxiaoya\n"),100);

HAL_UART_Receive()

以阻塞的方式从串口接收数据,设定指定时间,超过指定时间后不再接收此次数据,不产生中断

原型:HAL_StatusTypeDef HAL_UART_Receive(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size, uint32_t Timeout)

参数:
UART_HandleTypeDef *huart:串口句柄
uint8_t *pData:从串口接收的数据的存放地址
uint16_t Size:从串口接收的数据的大小(留一个字符存放 '\0' )
uint32_t Timeout:此次数据接收的时间

实例:HAL_UART_Receive(&huart1,myData,sizeof(myData) - 1,100);

使用STM32CubeMX创建工程

配置SYS

配置RCC

配置串口信息

配置工程名称、工程路径

选择固件库

生成工程

使用MicroLIB库

main函数编写

/* USER CODE BEGIN Header */
/**
  ******************************************************************************
  * @file           : main.c
  * @brief          : Main program body
  ******************************************************************************
  * @attention
  *
  * Copyright (c) 2023 STMicroelectronics.
  * All rights reserved.
  *
  * This software is licensed under terms that can be found in the LICENSE file
  * in the root directory of this software component.
  * If no LICENSE file comes with this software, it is provided AS-IS.
  *
  ******************************************************************************
  */
/* USER CODE END Header */
/* Includes ------------------------------------------------------------------*/
#include "main.h"
#include "usart.h"
#include "gpio.h"

/* Private includes ----------------------------------------------------------*/
/* USER CODE BEGIN Includes */

#include <string.h>
#include <stdio.h>

/* USER CODE END Includes */

/* Private typedef -----------------------------------------------------------*/
/* USER CODE BEGIN PTD */

/* USER CODE END PTD */

/* Private define ------------------------------------------------------------*/
/* USER CODE BEGIN PD */
/* USER CODE END PD */

/* Private macro -------------------------------------------------------------*/
/* USER CODE BEGIN PM */

/* USER CODE END PM */

/* Private variables ---------------------------------------------------------*/

/* USER CODE BEGIN PV */

/* USER CODE END PV */

/* Private function prototypes -----------------------------------------------*/
void SystemClock_Config(void);
/* USER CODE BEGIN PFP */

/* USER CODE END PFP */

/* Private user code ---------------------------------------------------------*/
/* USER CODE BEGIN 0 */

//重写stdio.c文件中的prinft()里的fputc()函数
int fputc(int my_data,FILE *p)
{
    unsigned char temp = my_data;
    //改写后,使用printf()函数会将数据通过串口1发送出去
    HAL_UART_Transmit(&huart1,&temp,1,0xffff);  //0xfffff为最大超时时间
    return my_data;
}
    
/* USER CODE END 0 */

/**
  * @brief  The application entry point.
  * @retval int
  */
int main(void)
{
  /* USER CODE BEGIN 1 */
    uint8_t myData[20] = {0};
  /* USER CODE END 1 */

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_USART1_UART_Init();
  /* USER CODE BEGIN 2 */

  //串口以阻塞的方式发送数据,超过时间后不再发送此次数据
  HAL_UART_Transmit(&huart1,(uint8_t *)"jiangxiaoya\n",strlen("jiangxiaoya\n"),100);  //单片机复位后,会发送一个数据

  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
    //串口以阻塞的方式接收数据,超过时间后不再接收此次数据
    HAL_UART_Receive(&huart1,myData,sizeof(myData) - 1,100);  
        
    printf("%s\r\n",myData);  //向串口发送数据
    memset(myData,0,sizeof(myData));
        
  }
  /* USER CODE END 3 */
}

/**
  * @brief System Clock Configuration
  * @retval None
  */
void SystemClock_Config(void)
{
  RCC_OscInitTypeDef RCC_OscInitStruct = {0};
  RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};

  /** Initializes the RCC Oscillators according to the specified parameters
  * in the RCC_OscInitTypeDef structure.
  */
  RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
  RCC_OscInitStruct.HSEState = RCC_HSE_ON;
  RCC_OscInitStruct.HSEPredivValue = RCC_HSE_PREDIV_DIV1;
  RCC_OscInitStruct.HSIState = RCC_HSI_ON;
  RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
  RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
  RCC_OscInitStruct.PLL.PLLMUL = RCC_PLL_MUL9;
  if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)
  {
    Error_Handler();
  }

  /** Initializes the CPU, AHB and APB buses clocks
  */
  RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK
                              |RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;
  RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;
  RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
  RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2;
  RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;

  if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_2) != HAL_OK)
  {
    Error_Handler();
  }
}

/* USER CODE BEGIN 4 */

/* USER CODE END 4 */

/**
  * @brief  This function is executed in case of error occurrence.
  * @retval None
  */
void Error_Handler(void)
{
  /* USER CODE BEGIN Error_Handler_Debug */
  /* User can add his own implementation to report the HAL error return state */
  __disable_irq();
  while (1)
  {
  }
  /* USER CODE END Error_Handler_Debug */
}

#ifdef  USE_FULL_ASSERT
/**
  * @brief  Reports the name of the source file and the source line number
  *         where the assert_param error has occurred.
  * @param  file: pointer to the source file name
  * @param  line: assert_param error line source number
  * @retval None
  */
void assert_failed(uint8_t *file, uint32_t line)
{
  /* USER CODE BEGIN 6 */
  /* User can add his own implementation to report the file name and line number,
     ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */
  /* USER CODE END 6 */
}
#endif /* USE_FULL_ASSERT */

疑难点分析

改写stdio.c中的fputc()函数,以便能使用printf()向串口发送数据

//重写stdio.c文件中的prinft()里的fputc()函数
int fputc(int my_data,FILE *p)
{
    unsigned char temp = my_data;
    //改写后,使用printf()函数会将数据通过串口一发送出去
    HAL_UART_Transmit(&huart1,&temp,1,0xffff);  //0xfffff为最大超时时间
    return my_data;
}

串口之中断法

串口数据接收的状态标记变量

该变量可以自行命名,是一个uint16_t 类型的变量如uint16_t UART1_RX_STA = 0。串口每接收到一个数据时,该变量就自增1,当串口接收到回车(0x0d和0x0a),即串口的缓存数据全部接收完成,让该变量的bit16置1,表示数据全部接收完成。

uint16_t UART1_RX_STA

bit 15

bit 14

bit 13~0

接收数据完成标志

接收0x0d标志

接收到有效数据个数标志

串口接收缓存和接收缓冲

  • 接收缓存:存放串口每次接收的数据(uint8_t buf=0;

  • 接收缓冲:存放接收缓存(uint8_t UART1_RX_Buffer[USART_REC_LEN] = {'\0'};

串口每接收一个字节就会调用一次串口接收数据完成回调函数(HAL_UART_RxCpltCallback()),因此,需要在串口接收数据完成回调函数中对串口每一次接收的数据进行处理,即把数据存放到接收缓冲中。

STM32的hal库关于串口中断的函数

HAL_UART_Transmit_IT()

原型:HAL_StatusTypeDef HAL_UART_Transmit_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)

参数:
UART_HandleTypeDef *huart:串口句柄
uint8_t *pData:向串口发送的数据的地址
uint16_t Size:向串口发送的数据的大小

HAL_UART_Receive_IT()

原型:HAL_StatusTypeDef HAL_UART_Receive_IT(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)

参数:
UART_HandleTypeDef *huart:串口句柄
uint8_t *pData:从串口接收的数据的存放地址
uint16_t Size:从串口接收的数据的大小(留一个字符存放 '\0' )

HAL_UART_TxCpltCallback()

串口发送数据完成回调函数,也就是串口发送一个字节数据会产生执行该函数,该函数是一个虚函数,我们可以在main.c文件中完成它的函数体

原型:__weak void HAL_UART_TxCpltCallback(UART_HandleTypeDef *huart)

参数:
UART_HandleTypeDef *huart:串口句柄

HAL_UART_RxCpltCallback()

串口接收数据完成回调函数,也就是串口接收一个字节数据都会产生中断执行该函数,该函数是一个虚函数,我们可以在main.c文件中完成它的函数体

原型:__weak void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)

参数:
UART_HandleTypeDef *huart:串口句柄

使用STM32CubeMX创建工程

配置SYS

配置RCC

配置串口信息

配置NVIC

配置工程名称、工程路径

选择固件库

生成工程

main函数编写

/* USER CODE BEGIN Header */
/**
  ******************************************************************************
  * @file           : main.c
  * @brief          : Main program body
  ******************************************************************************
  * @attention
  *
  * Copyright (c) 2023 STMicroelectronics.
  * All rights reserved.
  *
  * This software is licensed under terms that can be found in the LICENSE file
  * in the root directory of this software component.
  * If no LICENSE file comes with this software, it is provided AS-IS.
  *
  ******************************************************************************
  */
/* USER CODE END Header */
/* Includes ------------------------------------------------------------------*/
#include "main.h"
#include "usart.h"
#include "gpio.h"

/* Private includes ----------------------------------------------------------*/
/* USER CODE BEGIN Includes */

#include <stdio.h>  
#define USART_REC_LEN 200  //定义串口最大接收字符

/* USER CODE END Includes */

/* Private typedef -----------------------------------------------------------*/
/* USER CODE BEGIN PTD */

//串口接收缓存(1字节)
uint8_t buf = 0;

uint8_t UART1_RX_Buffer[USART_REC_LEN] = {'\0'};  //接收缓冲,串口接收的数据存放地点

//串口接收状态,16位
uint16_t UART1_RX_STA = 0;
//bit15: 如果是1表示接收完成
//bit14: 如果是1表示接收到回车(0x0d)
//bit13~bit0: 接收到的有效字节数目


/* USER CODE END PTD */

/* Private define ------------------------------------------------------------*/
/* USER CODE BEGIN PD */
/* USER CODE END PD */

/* Private macro -------------------------------------------------------------*/
/* USER CODE BEGIN PM */

/* USER CODE END PM */

/* Private variables ---------------------------------------------------------*/

/* USER CODE BEGIN PV */

/* USER CODE END PV */

/* Private function prototypes -----------------------------------------------*/
void SystemClock_Config(void);
/* USER CODE BEGIN PFP */

/* USER CODE END PFP */

/* Private user code ---------------------------------------------------------*/
/* USER CODE BEGIN 0 */

//串口接收回调函数,串口每接收到一个字节数据就会调用一次
void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{
    static int cnt = 0;
    
    //判断中断是否串口1触发的
    if(huart->Instance == USART1){
        
        //判断串口的缓存数据是否全部接收完成,即判断UART1_RX_STA的bit15是否为1
        if(!(UART1_RX_STA & 0x8000)){  //如果没接收完成
            
            //判断是否接收到回车:0x0d + 0x0a
            if(UART1_RX_STA & 0x4000){  //如果接收到0x0d
                
                //判断是否接收到0x0a
                if(buf == 0x0a){  //如果接收到0x0a,即此时接收到回车
                    
                    UART1_RX_STA |= 0x8000;  //串口的缓存数据全部接收完成
                
                }else{  //如果接收到0x0d没有接收到换行0x0a
                    cnt++;
                    UART1_RX_STA = 0;//则认为此次接收错误,串口重新开始接收下一字节数据
                    if(cnt == USART_REC_LEN){
                        UART1_RX_STA |= 0xbfff;   //如果收到0x0d后接收的100个字符内没有0x0a,就把UART1_RX_STA的bit14拉低,表示没有接收到0x0d
                    }
                }

            }else{  //如果没有接收到回车0x0d
                
                //则判断收到的这个字符是否是回车0x0d
                if(buf == 0x0d){  //如果该字符是0x0d
                    
                    UART1_RX_STA |= 0x4000;   //则将bit14拉高,表示接收到0x0d

                }else{  //如果不是0x0d
                    
                    //则将这个字符存放到缓存数组中
                    UART1_RX_Buffer[UART1_RX_STA & 0x3ffff] = buf;
                    UART1_RX_STA++;
                    
                    //如果接收的数据大于UART1_REC_LEN(200字节),则重新开始接收
                    if(UART1_RX_STA > USART_REC_LEN - 1){
                        UART1_RX_STA = 0;
                    }
                }
            }
        }
        cnt = 0;
        HAL_UART_Receive_IT(&huart1, &buf, 1);  //串口重新开启接收下一字节数据

    }
}

//重写stdio.h文件中的prinft()里的fputc()函数
int fputc(int my_data,FILE *p)
{
    unsigned char temp = my_data;
    //改写后,使用printf()函数会将数据通过串口一发送出去
    HAL_UART_Transmit(&huart1,&temp,1,0xffff);  //0xfffff为最大超时时间
    return my_data;
}

/* USER CODE END 0 */

/**
  * @brief  The application entry point.
  * @retval int
  */
int main(void)
{
  /* USER CODE BEGIN 1 */

  /* USER CODE END 1 */

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_USART1_UART_Init();
  /* USER CODE BEGIN 2 */

    //开启接收中断
    HAL_UART_Receive_IT(&huart1,&buf,1);  //开始接收串口的第一个字节,当接收到一个字节后就会产生中断进入接收回调函数

  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
        
        //判断判断串口是否接收完成
        if(UART1_RX_STA & 0x8000){
            printf("接收完成\r\n");
            
            //将接收到的数据发送到串口
            HAL_UART_Transmit(&huart1,UART1_RX_Buffer,UART1_RX_STA & 0x3fff,0xffff);
            
            //等待发送完成
            while(huart1.gState != HAL_UART_STATE_READY);
            
            //换行,重新开始下一次接收
            printf("\r\n");
            UART1_RX_STA = 0;
        }
        //心跳包
        printf("haozigegie1\r\n");
        HAL_Delay(1000);
  }
  /* USER CODE END 3 */
}

/**
  * @brief System Clock Configuration
  * @retval None
  */
void SystemClock_Config(void)
{
  RCC_OscInitTypeDef RCC_OscInitStruct = {0};
  RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};

  /** Initializes the RCC Oscillators according to the specified parameters
  * in the RCC_OscInitTypeDef structure.
  */
  RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
  RCC_OscInitStruct.HSEState = RCC_HSE_ON;
  RCC_OscInitStruct.HSEPredivValue = RCC_HSE_PREDIV_DIV1;
  RCC_OscInitStruct.HSIState = RCC_HSI_ON;
  RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
  RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
  RCC_OscInitStruct.PLL.PLLMUL = RCC_PLL_MUL9;
  if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)
  {
    Error_Handler();
  }

  /** Initializes the CPU, AHB and APB buses clocks
  */
  RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK
                              |RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;
  RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;
  RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
  RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2;
  RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;

  if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_2) != HAL_OK)
  {
    Error_Handler();
  }
}

/* USER CODE BEGIN 4 */

/* USER CODE END 4 */

/**
  * @brief  This function is executed in case of error occurrence.
  * @retval None
  */
void Error_Handler(void)
{
  /* USER CODE BEGIN Error_Handler_Debug */
  /* User can add his own implementation to report the HAL error return state */
  __disable_irq();
  while (1)
  {
  }
  /* USER CODE END Error_Handler_Debug */
}

#ifdef  USE_FULL_ASSERT
/**
  * @brief  Reports the name of the source file and the source line number
  *         where the assert_param error has occurred.
  * @param  file: pointer to the source file name
  * @param  line: assert_param error line source number
  * @retval None
  */
void assert_failed(uint8_t *file, uint32_t line)
{
  /* USER CODE BEGIN 6 */
  /* User can add his own implementation to report the file name and line number,
     ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */
  /* USER CODE END 6 */
}
#endif /* USE_FULL_ASSERT */

疑难点分析

fputc()函数分析

改写stdio.c中的fputc()函数,以便能使用printf()向串口发送数据

//重写stdio.c文件中的prinft()里的fputc()函数
int fputc(int my_data,FILE *p)
{
    unsigned char temp = my_data;
    //改写后,使用printf()函数会将数据通过串口一发送出去
    HAL_UART_Transmit(&huart1,&temp,1,0xffff);  //0xfffff为最大超时时间
    return my_data;
}

串口接收回调函数分析

当在main.c串口初始化后调用HAL_UART_Receive_IT()开始串口接收第一个字节数据,当接收到一个字节(即接收到接收缓存buf中)后就会产生中断进入接收回调函数对接收的数据进行处理

  • 如果该字节不是0x0d或0x0a则存到接收缓冲中

  • 如果该字节是0x0d,则串口数据接收的状态标记变量不再增加,等待接收到0x0a。

  1. 如果没有接收到0x0a则认为此次接收失败,则重新开始下一次接收。如果在接下来的100个字符内一直接收不到0x0a,则将UART1_RX_STA的bit14置0,表示没有接收到0x0d。

  1. 如果接收到了0x0a,则UART1_RX_STA的bit15置1表示数据全部接收完成,并等待该位被其他程序清除,重新开始下一次的串口接收。

  • 如果迟迟没有接收到0x0d,那么在接收字节超过USART_REC_LEN时,摈弃前面的数据,重新开始接收数据。

//串口接收回调函数,串口每接收到一个字节数据就会调用一次
void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{
    
    static int cnt = 0;
    
    //判断中断是否串口1触发的
    if(huart->Instance == USART1){
        
        //判断串口的缓存数据是否全部接收完成,即判断UART1_RX_STA的bit15是否为1
        if(!(UART1_RX_STA & 0x8000)){  //如果没接收完成
            
            //判断是否接收到回车:0x0d + 0x0a
            if(UART1_RX_STA & 0x4000){  //如果接收到0x0d
                
                //判断是否接收到0x0a
                if(buf == 0x0a){  //如果接收到0x0a,即此时接收到回车
                    
                    UART1_RX_STA |= 0x8000;  //串口的缓存数据全部接收完成
                
                }else{  //如果接收到0x0d没有接收到换行0x0a
                    cnt++;
                    UART1_RX_STA = 0;//则认为此次接收错误,串口重新开始接收下一字节数据
                    if(cnt == USART_REC_LEN){
                        UART1_RX_STA |= 0xbfff;   //如果收到0x0d后接收的100个字符内没有0x0a,就把UART1_RX_STA的bit14拉低,表示没有接收到0x0d
                    }
                }

            }else{  //如果没有接收到回车0x0d
                
                //则判断收到的这个字符是否是回车0x0d
                if(buf == 0x0d){  //如果该字符是0x0d
                    
                    UART1_RX_STA |= 0x4000;   //则将bit14拉高,表示接收到0x0d

                }else{  //如果不是0x0d
                    
                    //则将这个字符存放到缓存数组中
                    UART1_RX_Buffer[UART1_RX_STA & 0x3ffff] = buf;
                    UART1_RX_STA++;
                    
                    //如果接收的数据大于UART1_REC_LEN(200字节),则重新开始接收
                    if(UART1_RX_STA > USART_REC_LEN - 1){
                        UART1_RX_STA = 0;
                    }
                }
            }
        }
        cnt = 0;
        HAL_UART_Receive_IT(&huart1, &buf, 1);  //串口重新开启接收下一字节数据

    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/386562.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Vue基础入门讲义(三)-指令

文章目录1.什么是指令&#xff1f;2.插值表达式2.1.花括号2.2.插值闪烁2.3.v-text和v-html3.v-model4.v-on4.1.基本用法4.2.事件修饰5.v-for5.1.遍历数组5.2.数组角标5.3.遍历对象6.key7.v-if和v-show7.1.基本使用7.2.与v-for结合7.3.v-else7.4.v-show8.v-bind8.1. 属性上使用v…

服务器处理发生异常:java.text.ParseException: Unparseable date

测试上传报文的时候遇见报错 服务器处理发生异常:java.text.ParseException: Unparseable date: “2023/03/03” 错误报文 实际需要的报文 错误原因 上传时间字段&#xff0c;与Date字段数据位数不匹配&#xff0c;Java类型&#xff1a;Date默认带有年月日 时分秒yyyy-mm-dd…

十年业务开发总结,如何做好高效高质量的价值交付

作者&#xff1a;杨博林 阿里大淘宝场景金融团队 软件交付是一个非常复杂的过程和体系&#xff0c;需要保障好每个阶段的质量和效率才能保障最终的质量和效率。本文将尝试从需求交付的前、中、后三个环节来阐述一下如何做高效高质量的价值交付。 一、背景 转眼间已经做了十年的…

JavaScript基础内容

日升时奋斗&#xff0c;日落时自省 目录 1、基础语法 2、DOM 2.1、选中页面元素 2.2、获取/修改元素内容 3、JS案例 3.1、网页版本猜数字 3.2、网页版表白墙 JS最初只是为了进行前端页面的开发后来JS也被赋予了更多的功能&#xff0c;可以用来开发桌面程序&#xff0c;手…

RHCSA-重置root密码(3.3)

方法1&#xff1a;rd.break &#xff08;1&#xff09;首先重启系统&#xff0c;在此页面按e键&#xff0c;在屏幕上显示内核启动参数 &#xff08;2&#xff09;知道linux这行&#xff0c;末尾空格后输入rd.break&#xff0c;然后按ctrlx &#xff08;3&#xff09;查看&#…

电脑桌面上的图标不见了怎么办?5个完美的解决技巧

案例&#xff1a;电脑桌面不显示任何东西&#xff1f; “救命&#xff01;电脑打开后&#xff0c;只有桌面&#xff0c;任何图标都没有怎么办&#xff1f;心急&#xff0c;不知道该怎么解决&#xff1f;” 电脑桌面上的图标消失是一个比较常见的问题&#xff0c;许多用户都会…

Hadoop集群启动从节点没有DataNode

目录 一、问题背景 二、解决思路 三、解决办法&#xff1a; 一、问题背景 之前启动hadoop集群的时候都没有问题&#xff0c;今天启动hadoop集群的时候&#xff0c;从节点的DataNode没有启动起来。 二、解决思路 遇见节点起不来的情况&#xff0c;可以去看看当前节点的日志…

各大加密算法对比(原理、性能、安全、运用)

原理按加密可逆可以分为&#xff1a;加密可逆算法和加密不可逆算法。加密可逆算法又可以分为&#xff1a;对称加密和非对称加密。1、加密不可逆算法&#xff1a;一般采用hash算法加密&#xff0c;其原理一般是将原文长度补位成64的倍数&#xff0c;接着初始化固定长度的缓存值&…

大数据框架之Hadoop:MapReduce(五)Yarn资源调度器

Apache YARN (Yet Another Resource Negotiator) 是 hadoop 2.0 引入的集群资源管理系统。用户可以将各种服务框架部署在 YARN 上&#xff0c;由 YARN 进行统一地管理和资源分配。 简言之&#xff0c;Yarn是一个资源调度平台&#xff0c;负责为运算程序提供服务器运算资源&…

Windows Cannot Initialize Data Bindings 问题的解决方法

前言 拿到一个调试程序, 怎么折腾都打不开, 在客户那边, 尝试了几个系统版本, 发现Windows 10 21H2 版本可以正常运行。 尝试 系统篇 系统结果公司电脑 Windows 8有问题…下载安装 Windows10 22H2问题依旧下载安装 Windows10 21H2问题依旧家里的 笔记本Window 11正常 网上…

第三章 opengl之着色器

OpenGL着色器GLSLGLSL的数据类型向量输入与输出Uniform更多属性自己的着色器类着色器 着色器是运行在GPU上的小程序。着色器只是一种把输入转化为输出的程序。着色器也是一种非常独立的程序&#xff0c;因为它们之间不能相互通信&#xff1b;它们之间唯一的沟通只有通过输入和…

DC-DC模块电源隔离直流升压高压稳压输出5v12v24v转60v100v110v150v220v250v300v400v500v

特点效率高达80%以上1*1英寸标准封装单电压输出稳压输出工作温度: -40℃~85℃阻燃封装&#xff0c;满足UL94-V0 要求温度特性好可直接焊在PCB 上应用HRB 0.2~10W 系列模块电源是一种DC-DC升压变换器。该模块电源的输入电压分为&#xff1a;4.5~9V、9~18V、及18~36VDC标准&#…

软工第二次个人作业——软件案例分析

软工第二次个人作业——软件案例分析 项目内容这个作业属于哪个课程2023北航敏捷软件工程这个作业的要求在哪里个人作业-软件案例分析我在这个课程的目标是体验敏捷开发过程&#xff0c;掌握一些开发技能&#xff0c;为进一步发展作铺垫这个作业在哪个具体方面帮助我实现目标通…

四川大学软件学院|系统级编程期末复习

概述 选择题 50 分&#xff08;原题率 80%&#xff09;&#xff1a;http://tieba.baidu.com/p/1250021454?&share9105&frsharewise&unique967707B1DAECEF4A785B61D29AF36950&st1639102957&client_type1&client_version12.10.1&sfccopy&share…

使用C语言实现简单的PNG图像读取

概述 首先&#xff0c;关于png图像的结构&#xff1a;PNG文件的结构、PNG格式的数据结构。这两篇文章说的比较细。我简单地说一下我使用到的地方&#xff1a; 注&#xff1a;①引于PNG格式的数据结构。②引于PNG文件的结构 “png文件的前8个字节为固定的文件头信息&#xff0…

Ubuntu中安装StaMPS

Ubuntu中安装StaMPS0 StaMPS简介1 首先安装好MATLAB&#xff0c;安装一些依赖工具包2 安装StaMPS2.1 下载StaMPS安装包2.2 安装2.3 配置环境2.4 matlab中的路径设置0 StaMPS简介 官网&#xff1a;https://homepages.see.leeds.ac.uk/~earahoo/stamps/ A software package to e…

水浒英雄为啥输?因为没愿景

水浒英雄为啥输&#xff1f;输在没愿景&#xff01; 宋江没学企业管理&#xff0c;太遗憾 企业文化核心三件套&#xff1a;使命愿景价值观 趣讲大白话&#xff1a;没有远见成不了大事 【安志强趣讲信息科技92期】 ******************************* 水浒英雄的使命&#xff1a;替…

HTML的表格标签和列表标签

&#x1f31f;所属专栏&#xff1a;HTML只因变凤凰之路&#x1f414;作者简介&#xff1a;rchjr——五带信管菜只因一枚&#x1f62e;前言&#xff1a;该系列将持续更新HTML的相关学习笔记&#xff0c;欢迎和我一样的小白订阅&#xff0c;一起学习共同进步~&#x1f449;文章简…

如何成为一名黑客?基础入门

如何成为一名黑客&#xff1f;基础入门 相信大家对黑客一词并不陌生&#xff0c;因为从小便受电影的熏陶&#xff0c;黑客轻易就能攻入别人的系统也让不少人都羡慕不已&#xff0c;但是真正能够成为黑客的人少之又少。很多人都是三天打鱼两天晒网&#xff0c;学习进度不明显&a…

写入性能:TDengine 最高达到 InfluxDB 的 10.3 倍,TimeScaleDB 的 6.74 倍

上周三&#xff0c;TDengine 正式发布了基于 TSBS 的时序数据库&#xff08;Time Series Database&#xff0c;TSDB&#xff09;性能基准测试报告&#xff0c;该报告采用 TSBS 平台中针对 DevOps 的场景作为基础数据集&#xff0c;在相同的 AWS 云环境下对 TDengine 3.0、Times…