MCU 串口接收環形緩沖區的實現

環形緩沖區

1. 環形緩沖區的特性

1.先進先出

2. 當緩沖區被使用完,且又有新的數據需要存儲時,丟掉歷史最久的數據,保存最新的數據

現實中的存儲介質都是線性的,因此我們需要做一下處理,才能在功能上實現環形緩沖區

算法說明:
1、pHead和pTail分別是連續存儲介質的首地址和尾地址
2、pTail - pHead 的值是環形緩沖區的總長度
3、pValid 是使用區域的起始指針,取數據時的起點,當取數據時pValid要發生偏移
4、pValidTail 是使用區域的的結尾指針,存數據時的起點,當存數據時,pValidTail要發生偏移
5、現有長度為addLen字節要存入,當pValidTail + addLen > pTail 時(超出了緩沖區,這時就要繞到開頭pHead)
int len1 = pTail - pValidTail;
int len2 = addLen - len1;
pValidTail = pHead + len2;//新的使用區的尾指針
6、判斷總長度是否變更,即是否有數據覆蓋pValid所指向的區域,如果有,要偏移pValid

2. 算法驗證代碼

ringBuffer.c:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <assert.h>
#include "ringBuffer.h"
#define BUFFER_SIZE  16   //緩沖區的長度,可以修改static u32 validLen;//已使用的數據長度
static u8* pHead = NULL;//環形存儲區的首地址
static u8* pTail = NULL;//環形存儲區的結尾地址
static u8* pValid = NULL;//已使用的緩沖區的首地址
static u8* pValidTail = NULL;//已使用的緩沖區的尾地址/** 初始化環形緩沖區* 環形緩沖區這里可以是malloc申請的內存,也可以是Flash存儲介質* */
void initRingbuffer(void)
{if(pHead == NULL){pHead = (u8*) malloc(BUFFER_SIZE);}pValid = pValidTail = pHead;pTail = pHead + BUFFER_SIZE;validLen = 0;
}/** function:向緩沖區中寫入數據* param:@buffer 寫入的數據指針* 		 @addLen 寫入的數據長度* return:-1:寫入長度過大* 		  -2:緩沖區沒有初始化* */
int wirteRingbuffer(u8* buffer,u32 addLen)
{if(addLen > BUFFER_SIZE) return -2;if(pHead==NULL) return -1;assert(buffer);//將要存入的數據copy到pValidTail處if(pValidTail + addLen > pTail)//需要分成兩段copy{int len1 = pTail - pValidTail;int len2 = addLen - len1;memcpy( pValidTail, buffer, len1);memcpy( pHead, buffer + len1, len2);pValidTail = pHead + len2;//新的有效數據區結尾指針}else{memcpy( pValidTail, buffer, addLen);pValidTail += addLen;//新的有效數據區結尾指針}//需重新計算已使用區的起始位置if(validLen + addLen > BUFFER_SIZE){int moveLen = validLen + addLen - BUFFER_SIZE;//有效指針將要移動的長度if(pValid + moveLen > pTail)//需要分成兩段計算{int len1 = pTail - pValid;int len2 = moveLen - len1;pValid = pHead + len2;}else{pValid = pValid + moveLen;}validLen = BUFFER_SIZE;}else{validLen += addLen;}return 0;
}/** function:從緩沖區內取出數據* param   :@buffer:接受讀取數據的buffer*		    @len:將要讀取的數據的長度* return  :-1:沒有初始化*	 	    >0:實際讀取的長度* */
int readRingbuffer(u8* buffer,u32 len)
{if(pHead==NULL) return -1;assert(buffer);if(validLen ==0) return 0;if( len > validLen) len = validLen;if(pValid + len > pTail)//需要分成兩段copy{int len1 = pTail - pValid;int len2 = len - len1;memcpy( buffer, pValid, len1);//第一段memcpy( buffer+len1, pHead, len2);//第二段,繞到整個存儲區的開頭pValid = pHead + len2;//更新已使用緩沖區的起始}else{memcpy( buffer, pValid, len);pValid = pValid +len;//更新已使用緩沖區的起始}validLen -= len;//更新已使用緩沖區的長度return len;
}/** function:獲取已使用緩沖區的長度* return  :已使用的buffer長度* */
u32 getRingbufferValidLen(void)
{return validLen;
}/** function:釋放環形緩沖區* */
void releaseRingbuffer(void)
{if(pHead!=NULL) free(pHead);pHead = NULL;
}

ringBuffer.h

#ifndef RINGBUFFER_H_
#define RINGBUFFER_H_
typedef unsigned char u8;
typedef unsigned int u32;void initRingbuffer(void);
int wirteRingbuffer(u8* buffer,u32 len);
int readRingbuffer(u8* buffer,u32 len);
u32 getRingbufferValidLen(void);
void releaseRingbuffer(void);#endif /* RINGBUFFER_H_ */

?測試 main 函數

#include <stdio.h>
#include <stdlib.h>
#include "ringBuffer.h"
// 主函數
int main()
{char c;int readLen;u8 readBuffer[10];//setvbuf(stdout,NULL,_IONBF,0); //pinrtf、putchar不能立馬輸出,打開此注釋initRingbuffer();printf("Please enter a line [blank line to terminate]> ");do{c=getchar();putchar(c);switch(c){case 'Q':goto exit;break;case 'R':readLen = readRingbuffer(readBuffer,10);printf("readRingbuffer len:%d\n",readLen);if(readLen > 0){printf("readRingbuffer:");for(int i=0;i<readLen;i++){printf("%c ",(char)readBuffer[i]);}printf("\n");}break;default :if(c!='\n') wirteRingbuffer((u8*)&c,1);break;}}while (1);exit:releaseRingbuffer();printf("exit.\n");return 0;
}

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/716905.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/716905.shtml
英文地址,請注明出處:http://en.pswp.cn/news/716905.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

[計算機網絡]--I/O多路轉接之poll和epoll

前言 作者&#xff1a;小蝸牛向前沖 名言&#xff1a;我可以接受失敗&#xff0c;但我不能接受放棄 如果覺的博主的文章還不錯的話&#xff0c;還請點贊&#xff0c;收藏&#xff0c;關注&#x1f440;支持博主。如果發現有問題的地方歡迎?大家在評論區指正 目錄 一、poll函…

web漏洞與規避

文章目錄 一、XSS 跨站腳本攻擊1.1 XSS攻擊的主要類型反射型XSS存儲型XSSDOM型XSS 1.2 前端開發如何應對XSS 二、CSRF 跨站請求偽造2.1 CSRF例子2.2 前端開發如何應對CSRF 三、SQL 注入3.1 前端如何防御SQL注入 四、前端如何使用CSP 一、XSS 跨站腳本攻擊 攻擊者通過在受害者的…

stm32學習筆記:IIC通信(未完)

概述 第一塊&#xff1a;介紹協議規則&#xff0c;然后用軟件模擬的形式來實現協議。 第二塊&#xff1a;介紹STM32的iic外設&#xff0c;然后用硬件來實現協議。 程序一現象&#xff1a;通過軟件I2C通信&#xff0c;對MPU6050芯片內部的寄存器進行讀寫&#xff0c;寫入到配…

【Synchronized同步原理】

Synchronized同步原理 1. synchronized的使用&#xff1f;2. 如何保證線程安全的&#xff1f;3.可重入原理&#xff08;加鎖次數計數器&#xff09;4. 原子性和可見性(順序性) 1. synchronized的使用&#xff1f; 對象鎖方法鎖類鎖 2. 如何保證線程安全的&#xff1f; publi…

關于數據提交上傳服務端的數據類型以及項目打包上線的流程

1 請求頭的類型&#xff1a; content-type&#xff1b; 01: application/json 數據以json格式請求&#xff1a;{"key":"value"} 02: application/x-www.form-urlencoded from表單的數據格式 name"zs"&age12 03 mutipart/form-data…

重學SpringBoot3-自動配置機制

重學SpringBoot3-自動配置機制 引言Spring Boot 自動配置原理示例&#xff1a;Spring Boot Web 自動配置深入理解總結相關閱讀 引言 Spring Boot 的自動配置是其最強大的特性之一&#xff0c;它允許開發者通過最少的配置實現應用程序的快速開發和部署。這一切都得益于 Spring …

飛槳(PaddlePaddle)模型保存與加載教程

文章目錄 飛槳&#xff08;PaddlePaddle&#xff09;模型保存與加載教程1. 概述2. 訓練調優場景2.1 保存動態圖模型2.2 加載動態圖模型 3. 推理部署場景3.1 使用基礎API 4. 其他場景4.1 舊版本格式兼容載入4.2 靜態圖模型的保存與加載 5. 總結 飛槳&#xff08;PaddlePaddle&am…

OmniPlan Pro mac版:簡單、智能,項目管理新選擇!

OmniPlan Pro是一款功能強大的項目管理軟件&#xff0c;它以其直觀的用戶界面和豐富的功能&#xff0c;幫助用戶輕松管理各種復雜的項目。無論是個人任務還是團隊協作&#xff0c;OmniPlan Pro都能提供全面的解決方案&#xff0c;讓項目管理變得更加簡單高效。 OmniPlan Pro軟…

從0開始回顧Mysql --- MySQL初體驗

大白話從0開始回顧MySQL&#xff0c;去除了一些繁瑣的操作的演示以及內容&#xff0c;如MySQL安裝等&#xff0c;本篇文章適合復習MySQL語法&#xff0c;學習MySQL語句&#xff0c;對MySQL不太熟練的同學&#xff0c;希望對大家有一些幫助。 MySQL初體驗 首先&#xff0c;我將…

如何在Window系統部署VisualSVN服務并結合cpolar實現無公網ip遠程訪問

文章目錄 前言1. VisualSVN安裝與配置2. VisualSVN Server管理界面配置3. 安裝cpolar內網穿透3.1 注冊賬號3.2 下載cpolar客戶端3.3 登錄cpolar web ui管理界面3.4 創建公網地址 4. 固定公網地址訪問 前言 SVN 是 subversion 的縮寫&#xff0c;是一個開放源代碼的版本控制系統…

數字孿生與智慧交通的融合發展:推動交通行業數字化轉型,構建智慧城市新生態

隨著信息技術的快速發展和城市化進程的深入推進&#xff0c;交通行業正面臨著前所未有的機遇與挑戰。傳統的交通管理模式已難以滿足日益增長的交通需求&#xff0c;而數字化轉型則成為了推動交通行業創新發展的必由之路。數字孿生技術作為一種前沿的信息技術手段&#xff0c;為…

聊一聊腦機接口

截至我所知的信息&#xff0c;腦機接口&#xff08;Brain-Computer Interface, BCI&#xff09;技術正在迅速發展&#xff0c;但仍處于相對早期階段。這一領域的研究涉及神經科學、工程學、計算機科學和人工智能等多個學科。以下是一些關于腦機接口目前進展的要點&#xff1a; …

基于springboot+vue的公寓報修管理系統

博主主頁&#xff1a;貓頭鷹源碼 博主簡介&#xff1a;Java領域優質創作者、CSDN博客專家、阿里云專家博主、公司架構師、全網粉絲5萬、專注Java技術領域和畢業設計項目實戰&#xff0c;歡迎高校老師\講師\同行交流合作 ?主要內容&#xff1a;畢業設計(Javaweb項目|小程序|Pyt…

每天的三哥牌小雞腿沒了

查看印度基金(164824)公告&#xff0c;從2024年2月28日起&#xff0c;暫停印度基金的申購&#xff0c;限額100塊的一拖七套利告一段落。這意味著看好印度股市的國內投資者&#xff0c;以后只能通過在場內買入獲得份額。 關于暫停申購的原因&#xff0c;官方說是為保護基金持有…

學Python如此簡單--停車管理模塊

簡介 &#xff08;本停車場一個小時4塊&#xff09; 模塊名稱&#xff1a;黑心停車管理模塊 模塊功能&#xff1a;汽車進場錄入、汽車出場收費 代碼 #進入停車場 def enter():print(歡迎進入黑心停車場)#模擬識別車牌號car_number input(請輸入車牌號:)t 0 #停車時間car…

重磅!交通領域頂級會議TRB會議將進行重大改革

美國交通研究委員會年會&#xff08;Transportation Research Board annual meeting,以下簡稱TRB會議&#xff09;是由美國交通研究委員會舉辦的交通領域的國際頂級會議。該會議每年舉辦一次&#xff0c;在華盛頓特區召開。TRB會議是交通研究領域知名度最高的學術會議之一&…

安裝 node 錯誤的配置環境變量之后使用 npm 報錯

安裝 node 錯誤的配置環境變量之后使用 npm 報錯 node:internal/modules/cjs/loader:1147 throw err; ^ Error: Cannot find module ‘F:\ACodeTools\Node\node_modules\npm\bin\node_modules\npm\bin\npm-cli.js’ at Module._resolveFilename (node:internal/modules/cjs/loa…

基于springboot實現企業員工績效考評系統項目【項目源碼+論文說明】

基于springboot實現企業員工績效考評系統演示 摘要 時代的變化速度實在超出人類的所料&#xff0c;21世紀&#xff0c;計算機已經發展到各行各業&#xff0c;各個地區&#xff0c;它的載體媒介-計算機&#xff0c;大眾稱之為的電腦&#xff0c;是一種特高速的科學儀器&#xf…

AI-數學-高中-32-統計-樣本空間與隨機事件

原作者視頻&#xff1a;【概率】【一數辭典】1樣本空間與隨機事件_嗶哩嗶哩_bilibili 1.隨機試驗&#xff1a; 2.樣本點、樣本空間、有限樣本空間&#xff1a; 示例1 示例2 3.事件&#xff1a; 示例&#xff1a;

[DEBUG] spring boot-如何處理鏈接中的空格等特殊字符

問題&#xff1a; get或者post中提交的內容可能有空格、#等特殊字符&#xff0c;不做處理的話可能解析錯誤。 解決&#xff1a; html中&#xff1a; <a th:href"{/listSgrna(id${item.getGeneId()},geneName${item.getGeneName()},genome${genome},sgrnaNum${sgrnaN…