数据接收处理之环形缓冲区

1 什么是环形缓冲区

环形缓冲区是一个先进先出(FIFO)的闭环的存储空间。通俗的理解为,在内存中规划了一块连续等分的环形内存,然后分别通过写指针和读指针对缓冲区的数据进行读写。而实际上,计算机内存的空间位置不一定是环形的,所以我们可以开辟一段连续的内存,然后将其首尾相连,如下图所示:
在这里插入图片描述
存储数据:当每有一个数据需要存储时,就存储在Write所指向的位置,存储完毕后Write将指向下一个位置。
读取数据:每当读取一个数据时,将读取Read指向的位置,当读取完毕后Read指向下一个位置。

2 环形缓冲区代码实现(基于C语言)

2.1 结构体定义

定义结构体类型 RingQueueBuffer_t 用于开辟环形缓冲区。

#define RING_QUEUE_BUFF_SIZE 1024   //配置缓冲区开辟大小
typedef struct ring_queue {
	uint16_t head;							//记录读位置,相当于Read
	uint16_t tail;							//记录写位置,相当于Write
	uint16_t lenth;							//记录缓冲区的数据个数
	uint8_t  buff[RING_QUEUE_BUFF_SIZE];	//缓冲区
}RingQueueBuffer_t;

2.2 缓冲区的初始化

初始化过程只需要将头指针、尾指针、缓冲区数据长度清零即可。

void RQBuff_Init(RingQueueBuffer_t* RQ_Buff) {
	RQ_Buff->head = RQ_Buff->tail = RQ_Buff->lenth = 0;
}

2.3 缓冲区添加数据

1.在添加数据的过程中需要保证缓冲区非满,通过检测缓冲区的数据长度是否等于所开辟缓冲区大小RING_QUEUE_BUFF_SIZE 即可。
2.添加数据:在缓冲区的tail写入要添加的数据即可。
3.添加数据完毕后,将tail指向下一个位置,并记录缓冲区的数据长度 lenth++
注意:(RQ_Buff->tail + 1) % RING_QUEUE_BUFF_SIZE 可以实现tail 指向末尾位置后自动跳转到缓冲区的0位置。

uint8_t RQBuff_AddData(RingQueueBuffer_t* RQ_Buff, uint8_t data) {
	///--溢出检测
	if (RQ_Buff->lenth >= RING_QUEUE_BUFF_SIZE) {
		return 0;
	}
	///---写入数据
	RQ_Buff->buff[RQ_Buff->tail] = data;
	///写指针位置+1,防溢出
	RQ_Buff->tail = (RQ_Buff->tail + 1) % RING_QUEUE_BUFF_SIZE;
	///数据长度计数
	RQ_Buff->lenth++;
	return 1;
}

2.4 读取数据

1.在读取数据的过程中需要保证缓冲区非空,通过检测缓冲区的数据长度是否等于0即可。
2.读取数据:读出缓冲区head指向的数据。
3.读取数据完毕后,将head指向下一个位置。由于读取后缓冲区数据少了一个,则记录缓冲区的数据长度 lenth–
注意:(RQ_Buff->head + 1) % RING_QUEUE_BUFF_SIZE 可以实现*head * 指向末尾位置后自动跳转到缓冲区的0位置。

uint8_t RQBuff_ReadData(RingQueueBuffer_t* RQ_Buff,uint8_t* data) {
	///--缓冲区空检测
	if (RQ_Buff->lenth == 0) {
		return 0;
	}
	///---读出数据
	*data = RQ_Buff->buff[RQ_Buff->head];//FIFO
	///读指针位置+1,防溢出
	RQ_Buff->head = (RQ_Buff->head + 1) % RING_QUEUE_BUFF_SIZE;
	///数据长度计数
	RQ_Buff->lenth--;
	return 1;
}

2.5 获取缓冲区数据长度

uint16_t RQBuff_GetBuffLenth(RingQueueBuffer_t* RQ_Buff) {
	return RQ_Buff->lenth;
}

3 环形缓冲区的完整代码

3.1 头文件

#ifndef __DATAPROCE_H
#define __DATAPROCE_H
#include <stdint.h>
/*
* 数据缓冲区配置 
*/
#define RING_QUEUE_BUFF_SIZE 1024   //配置缓冲区开辟大小
typedef struct ring_queue {
	uint16_t head;
	uint16_t tail;
	uint16_t lenth;
	uint8_t  buff[RING_QUEUE_BUFF_SIZE];
}RingQueueBuffer_t;
/*
  *
*/
void RQBuff_Init(RingQueueBuffer_t* RQ_Buff);						//初始化指定缓冲区
uint8_t RQBuff_AddData(RingQueueBuffer_t* RQ_Buff, uint8_t data);	//指定缓冲区添加数据
uint8_t RQBuff_ReadData(RingQueueBuffer_t* RQ_Buff, uint8_t* data);	//指定缓冲区读取数据
uint16_t RQBuff_GetBuffLenth(RingQueueBuffer_t* RQ_Buff);			//读取指定缓冲区的数据长度
#endif

3.2 源文件

#include "DataProce.h"
/**
  ==============================================================================
					 ##### 数据缓冲区系列函数 #####
  ==============================================================================
[..]
	(#)  初始化缓冲区函数:				RQBuff_Init()
	(#)	 添加数据使用函数:				RQBuff_AddData()
	(#)	 读取数据使用函数:				RQBuff_ReadData()
	(#)	 获得缓冲区数据长度使用函数:	RQBuff_GetBuffLenth()
		 (++)输入参数RQ_Buff类型为RingQueueBuffer_t,该参数作为被操作的对象,必须为
		 提前定义好的全局变量
  ******************************************************************************
  */
/**
* 函数:RQBuff_Init()
* 该函数初始化指定的缓冲区
* 输入参数:
*	RQ_Buff:待操作的缓冲区
* 返回值: 无
**/
void RQBuff_Init(RingQueueBuffer_t* RQ_Buff) {
	RQ_Buff->head = RQ_Buff->tail = RQ_Buff->lenth = 0;
}
/**
* 函数:RQBuff_AddData()
* 该函数向指定的缓冲区添加一个指定的u8数据
* 输入参数:
*	RQ_Buff:待操作的缓冲区
*	   data:需要添加的数据
* 返回值: 0:添加失败 1:添加成功
**/
uint8_t RQBuff_AddData(RingQueueBuffer_t* RQ_Buff, uint8_t data) {
	///--溢出检测
	if (RQ_Buff->lenth >= RING_QUEUE_BUFF_SIZE) {
		return 0;
	}
	///---写入数据
	RQ_Buff->buff[RQ_Buff->tail] = data;
	///写指针位置+1,防溢出
	RQ_Buff->tail = (RQ_Buff->tail + 1) % RING_QUEUE_BUFF_SIZE;
	///数据长度计数
	RQ_Buff->lenth++;
	return 1;
}
/**
* 函数:RQBuff_ReadData()
* 该函数从指定的缓冲区读取一个数据
* 输入参数:
*	RQ_Buff:待操作的缓冲区
*	   data:读取数据的存储地址
* 返回值: 0:读取失败 1:读取成功
**/
uint8_t RQBuff_ReadData(RingQueueBuffer_t* RQ_Buff,uint8_t* data) {
	///--缓冲区空检测
	if (RQ_Buff->lenth == 0) {
		return 0;
	}
	///---读出数据
	*data = RQ_Buff->buff[RQ_Buff->head];//FIFO
	///读指针位置+1,防溢出
	RQ_Buff->head = (RQ_Buff->head + 1) % RING_QUEUE_BUFF_SIZE;
	///数据长度计数
	RQ_Buff->lenth--;
	return 1;
}
/**
* 函数:RQBuff_GetBuffLenth()
* 该函数获得指定缓冲区的数据长度
* 输入参数:
*	RQ_Buff:待操作的缓冲区
* 返回值: 缓冲区的数据长度
**/
uint16_t RQBuff_GetBuffLenth(RingQueueBuffer_t* RQ_Buff) {
	return RQ_Buff->lenth;
}

4 环形缓冲区使用方法

1.建立一个缓冲区

RingQueueBuffer_t RQBuff1;

2.初始化缓冲区

RQBuff_Init(&RQBuff1);

3.添加缓冲数据

/*其他得到数据的代码*/
RQBuff_AddData(&RQBuff1,data);

4.处理数据

if(RQBuff_GetBuffLenth(&RQBuff1)!=0){//缓冲区有数据
	/*读取数据,处理数据/
}
  • 6
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
STM32F4,可以使用DMA和断来实现串口环形缓冲区接收多字节数据。 一种常见的方法使用DMA,在接收数据时触发DMA传输,将数据存储到环形缓冲区。具体步骤如下: 1. 初始化串口和DMA,将DMA模式设置为循环模式。 2. 设置DMA的目的地址为环形缓冲区的当前位置,设置传输大小为1字节。 3. 开启串口接收断,并在断服务函数触发DMA传输。 4. 在DMA传输完成更新环形缓冲区指针,并重新配置DMA传输。 以下是一个示例代码,仅供参考: ```c #define BUFFER_SIZE 64 uint8_t buffer[BUFFER_SIZE]; volatile uint8_t head = 0; volatile uint8_t tail = 0; void USART2_IRQHandler(void) { if (USART_GetITStatus(USART2, USART_IT_RXNE) != RESET) { // 触发DMA传输 DMA_Cmd(DMA1_Stream5, ENABLE); } } void DMA1_Stream5_IRQHandler(void) { if (DMA_GetITStatus(DMA1_Stream5, DMA_IT_TCIF5) != RESET) { // 更新缓冲区指针 head = (head + 1) % BUFFER_SIZE; // 重新配置DMA传输 DMA_ClearITPendingBit(DMA1_Stream5, DMA_IT_TCIF5); DMA_Cmd(DMA1_Stream5, DISABLE); DMA_SetCurrDataCounter(DMA1_Stream5, 1); DMA_SetMemory0Address(DMA1_Stream5, (uint32_t)&buffer[head]); DMA_Cmd(DMA1_Stream5, ENABLE); } } int main(void) { // 初始化串口和DMA // 配置DMA DMA_DeInit(DMA1_Stream5); DMA_InitTypeDef DMA_InitStruct; DMA_StructInit(&DMA_InitStruct); DMA_InitStruct.DMA_Channel = DMA_Channel_4; DMA_InitStruct.DMA_PeripheralBaseAddr = (uint32_t)&USART2->DR; DMA_InitStruct.DMA_Memory0BaseAddr = (uint32_t)&buffer[head]; DMA_InitStruct.DMA_DIR = DMA_DIR_PeripheralToMemory; DMA_InitStruct.DMA_BufferSize = 1; DMA_InitStruct.DMA_PeripheralInc = DMA_PeripheralInc_Disable; DMA_InitStruct.DMA_MemoryInc = DMA_MemoryInc_Enable; DMA_InitStruct.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte; DMA_InitStruct.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte; DMA_InitStruct.DMA_Mode = DMA_Mode_Circular; DMA_InitStruct.DMA_Priority = DMA_Priority_High; DMA_InitStruct.DMA_FIFOMode = DMA_FIFOMode_Disable; DMA_InitStruct.DMA_FIFOThreshold = DMA_FIFOThreshold_HalfFull; DMA_InitStruct.DMA_MemoryBurst = DMA_MemoryBurst_Single; DMA_InitStruct.DMA_PeripheralBurst = DMA_PeripheralBurst_Single; DMA_Init(DMA1_Stream5, &DMA_InitStruct); // 开启串口接收断 USART_ITConfig(USART2, USART_IT_RXNE, ENABLE); // 启动DMA传输 DMA_Cmd(DMA1_Stream5, ENABLE); while (1) { // 处理接收到的数据 if (head != tail) { // 读取缓冲区数据 uint8_t data = buffer[tail]; // 更新缓冲区指针 tail = (tail + 1) % BUFFER_SIZE; // 处理数据 // ... } } } ``` 在上面的代码,head和tail分别表示环形缓冲区的头部和尾部。当接收数据时,触发DMA传输将数据存储到缓冲区的当前位置。当DMA传输完成时,更新头部指针,并重新配置DMA传输。在主循环,读取缓冲区数据处理。由于环形缓冲区是循环的,当头部指针超过缓冲区末尾时,会自动返回到缓冲区开头。
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值