当前位置: 首页 > news >正文

广东微信网站制作报价沈阳h5网站建设

广东微信网站制作报价,沈阳h5网站建设,上海建设网站的公司,dede英文网站本期主题#xff1a; 讲解ARRAY_SIZE的作用以及定义#xff0c;还有一个踩坑分析 往期链接#xff1a; 数据结构系列——先进先出队列queue数据结构系列——栈 stackLinux内核链表零长度数组的使用inline的作用嵌入式C基础——ARRAY_SIZE使用以及踩坑分析 目录 1. Ringbuff…本期主题 讲解ARRAY_SIZE的作用以及定义还有一个踩坑分析 往期链接 数据结构系列——先进先出队列queue数据结构系列——栈 stackLinux内核链表零长度数组的使用inline的作用嵌入式C基础——ARRAY_SIZE使用以及踩坑分析 目录 1. Ringbuffer定义及特点2.ringbuffer实例rtos实例2.1 ringbuffer结构体2.2 ringbuffer获取当前长度2.3 ringbuffer put 3.根据rtos代码设计ringbuffer并测试 1. Ringbuffer定义及特点 Ringbuffer的定义 环形缓冲区Ring Buffer也称为循环缓冲区、环形队列Ring Queue或循环队列Circular Queue是一种用于在固定大小的存储区域中存储数据的数据结构。 环形缓冲区通常由一个固定大小的数组和两个指针组成分别用于指示缓冲区的起始位置和结束位置。数据被顺序地存储在数组中当到达数组的末尾时数据会“循环”回到数组的起始位置实现了环形的存储结构。 Ringbuffer的特点 环形缓冲区通常用于实现数据在生产者和消费者之间的高效传输特别是在多线程或多任务环境中。它具有以下特点和优势 高效性 环形缓冲区采用循环结构避免了数据的频繁搬移。这使得对于生产者和消费者来说插入和删除操作的时间复杂度都是 O(1)。固定大小 环形缓冲区有一个固定的大小这使得其占用的内存是可控的。当缓冲区已满时生产者会被阻塞以免过度生产数据同时保护消费者不会因为数据积压而失去响应能力。无需动态内存分配 环形缓冲区一般使用静态数组作为存储空间无需动态内存分配因此可以在嵌入式系统等资源受限的环境中使用。循环利用空间 环形缓冲区的循环结构使得空间可以被循环利用即使数据已经被消费存储空间也可以被后续的数据重新利用。并发安全 当环形缓冲区被多个线程或任务访问时需要通过互斥锁或其他同步机制来保护共享资源以确保线程安全。 使用环形缓冲区时需要注意处理好生产者和消费者之间的同步和竞态条件以及处理好缓冲区空间不足和溢出的情况。同时还需要考虑如何优雅地处理缓冲区已满和已空时的阻塞与唤醒机制。 2.ringbuffer实例rtos实例 ringbuffer比较难处理的一个问题就是 当read_ptr write_ptr 时此时并不是很确定ringbuffer是empty还是fullrtos和Linux中的设计使用的是 镜像法具体的意思是 镜像扩展位是环形缓冲区中的一个额外的标志位或变量用于指示缓冲区的填充状态。它通常有两种取值 未填满Not Full 表示缓冲区还有空闲空间可以继续向缓冲区写入数据。填满Full 表示缓冲区已经填满不能再继续向缓冲区写入数据。此时继续写入数据可能会覆盖已有的数据造成数据丢失。 使用镜像扩展位当缓冲区填满时生产者可以根据镜像扩展位的状态来判断是否继续写入数据。如果缓冲区未填满生产者可以继续写入数据如果缓冲区已经填满则生产者应该等待直到缓冲区有空间可用。 2.1 ringbuffer结构体 看rtos的实例代码 struct rt_ringbuffer {rt_uint8_t *buffer_ptr;/* use the msb of the {read,write}_index as mirror bit. You can see this as* if the buffer adds a virtual mirror and the pointers point either to the* normal or to the mirrored buffer. If the write_index has the same value* with the read_index, but in a different mirror, the buffer is full.* While if the write_index and the read_index are the same and within the* same mirror, the buffer is empty. The ASCII art of the ringbuffer is:** mirror 0 mirror 1* ---------------------|~~~~~~~~~~~~~~~~~~~~~* | 0 | 1 | 2 | 3 | 4 | 5 | 6 ||| 0 | 1 | 2 | 3 | 4 | 5 | 6 | Full* ---------------------|~~~~~~~~~~~~~~~~~~~~~* read_idx-^ write_idx-^** ---------------------|~~~~~~~~~~~~~~~~~~~~~* | 0 | 1 | 2 | 3 | 4 | 5 | 6 ||| 0 | 1 | 2 | 3 | 4 | 5 | 6 | Empty* ---------------------|~~~~~~~~~~~~~~~~~~~~~* read_idx-^ ^-write_idx*/rt_uint32_t read_mirror : 1;rt_uint32_t read_index : 31;rt_uint32_t write_mirror : 1;rt_uint32_t write_index : 31;/* as we use msb of index as mirror bit, the size should be signed and* could only be positive. */rt_int32_t buffer_size; };结构体中有一个read_mirror和write_mirror看注释可以知道是表示read/write的指针是否到了mirror区并且 read_index write_index read_mirror和write_mirror不同此时为FULLread_index write_index read_mirror和write_mirror相同此时为EMPTY 2.2 ringbuffer获取当前长度 RTOS中代码 rt_size_t rt_ringbuffer_data_len(struct rt_ringbuffer *rb) {switch (rt_ringbuffer_status(rb)){case RT_RINGBUFFER_EMPTY:return 0;case RT_RINGBUFFER_FULL:return rb-buffer_size;case RT_RINGBUFFER_HALFFULL:default:{rt_size_t wi rb-write_index, ri rb-read_index;if (wi ri)return wi - ri;elsereturn rb-buffer_size - (ri - wi);}} }有三种情况 当rt_status为空时len为0当rt_status为满时len为buffer_size当ringbuffer非空/满时如果write_indexread_index那么buffer_size为 wi-ri否则为 rb-buffer_size - (ri - wi) 2.3 ringbuffer put put代码如下 rt_size_t rt_ringbuffer_put(struct rt_ringbuffer *rb,const rt_uint8_t *ptr,rt_uint32_t length) {rt_uint32_t size;RT_ASSERT(rb ! RT_NULL);/* whether has enough space */size rt_ringbuffer_space_len(rb);/* no space */if (size 0)return 0;/* drop some data */if (size length)length size;if (rb-buffer_size - rb-write_index length){/* read_index - write_index empty space */rt_memcpy(rb-buffer_ptr[rb-write_index], ptr, length);/* this should not cause overflow because there is enough space for* length of data in current mirror */rb-write_index length;return length;}rt_memcpy(rb-buffer_ptr[rb-write_index],ptr[0],rb-buffer_size - rb-write_index);rt_memcpy(rb-buffer_ptr[0],ptr[rb-buffer_size - rb-write_index],length - (rb-buffer_size - rb-write_index));/* we are going into the other side of the mirror */rb-write_mirror ~rb-write_mirror;rb-write_index length - (rb-buffer_size - rb-write_index);return length; } 逻辑是 判断剩余空间的size是否足够放下数据如果不够则需要drop一些数据如果可以直接放不需要循环回来那么就直接Memcpy就行如果需要循环回来write_mirror需要取反 3.根据rtos代码设计ringbuffer并测试 代码如下 #include stdio.h #include stdlib.h #include unistd.h #include pthread.h #include semaphore.h #include string.h #include main.hstruct rt_ringbuffer {rt_uint8_t *buffer_ptr;rt_uint32_t read_mirror : 1;rt_uint32_t read_index : 31;rt_uint32_t write_mirror : 1;rt_uint32_t write_index : 31;rt_int32_t buffer_size; };enum rt_ringbuffer_state rt_ringbuffer_status(struct rt_ringbuffer *rb) {if (rb-read_index rb-write_index){if (rb-read_mirror rb-write_mirror)return RT_RINGBUFFER_EMPTY;elsereturn RT_RINGBUFFER_FULL;}return RT_RINGBUFFER_HALFFULL; }int rt_ringbuffer_data_len(struct rt_ringbuffer *rb) {switch (rt_ringbuffer_status(rb)){case RT_RINGBUFFER_EMPTY:return 0;case RT_RINGBUFFER_FULL:return rb-buffer_size;case RT_RINGBUFFER_HALFFULL:default:{rt_size_t wi rb-write_index, ri rb-read_index;if (wi ri)return wi - ri;elsereturn rb-buffer_size - (ri - wi);}} }struct rt_ringbuffer *ringbuffer_init(rt_int32_t size) {struct rt_ringbuffer *rb;rt_uint8_t *pool;rb (struct rt_ringbuffer *)malloc(sizeof(struct rt_ringbuffer));if (!rb) {printf(rb is null\n);return NULL;}pool (rt_uint8_t *)malloc(size);if (!pool) {printf(pool is null\n);return NULL;}rb-buffer_ptr pool;rb-buffer_size size;rb-read_mirror rb-read_index 0;rb-write_mirror rb-write_index 0;return rb; } /** return the size of empty space in rb */ #define rt_ringbuffer_space_len(rb) ((rb)-buffer_size - rt_ringbuffer_data_len(rb))int rt_ringbuffer_put(struct rt_ringbuffer *rb,const rt_uint8_t *ptr,rt_uint32_t length) {rt_uint32_t size;/* whether has enough space */size rt_ringbuffer_space_len(rb);/* no space */if (size 0)return 0;/* drop some data */if (size length)length size;if (rb-buffer_size - rb-write_index length){/* read_index - write_index empty space */memcpy(rb-buffer_ptr[rb-write_index], ptr, length);/* this should not cause overflow because there is enough space for* length of data in current mirror */rb-write_index length;return length;}memcpy(rb-buffer_ptr[rb-write_index],ptr[0],rb-buffer_size - rb-write_index);memcpy(rb-buffer_ptr[0],ptr[rb-buffer_size - rb-write_index],length - (rb-buffer_size - rb-write_index));/* we are going into the other side of the mirror */rb-write_mirror ~rb-write_mirror;rb-write_index length - (rb-buffer_size - rb-write_index);return length; }int rt_ringbuffer_get(struct rt_ringbuffer *rb,rt_uint8_t *ptr,rt_uint32_t length) {int size;/* whether has enough data */size rt_ringbuffer_data_len(rb);/* no data */if (size 0)return 0;/* less data */if (size length)length size;if (rb-buffer_size - rb-read_index length){/* copy all of data */memcpy(ptr, rb-buffer_ptr[rb-read_index], length);/* this should not cause overflow because there is enough space for* length of data in current mirror */rb-read_index length;return length;}memcpy(ptr[0],rb-buffer_ptr[rb-read_index],rb-buffer_size - rb-read_index);memcpy(ptr[rb-buffer_size - rb-read_index],rb-buffer_ptr[0],length - (rb-buffer_size - rb-read_index));/* we are going into the other side of the mirror */rb-read_mirror ~rb-read_mirror;rb-read_index length - (rb-buffer_size - rb-read_index);return length; }int main(void) {int ret;struct rt_ringbuffer *rb NULL;rb ringbuffer_init(4);if (!rb) {printf(rb init failed\n);return -1;}uint8_t a[4] {0x1, 0x2, 0x3, 0x4};uint8_t read[4] { 0 };ret rt_ringbuffer_put(rb, a, 4);ret rt_ringbuffer_get(rb, read, 1);for (int i 0; i 4; i) {printf(0x%x\n, read[i]);}ret rt_ringbuffer_put(rb, a, 4);ret rt_ringbuffer_get(rb, read, 4);for (int i 0; i 4; i) {printf(0x%x\n, read[i]);}return 0; }//main.h #ifndef __MAIN_H__ #define __MAIN_H__#include stdio.h #include stdint.htypedef uint8_t rt_uint8_t; typedef uint32_t rt_uint32_t; typedef int32_t rt_int32_t; typedef int rt_size_t;enum rt_ringbuffer_state {RT_RINGBUFFER_EMPTY,RT_RINGBUFFER_FULL,/* half full is neither full nor empty */RT_RINGBUFFER_HALFFULL, };#endif测试结果
http://www.laogonggong.com/news/119956.html

相关文章:

  • 浙江省住房和城乡建设厅网站打不开食品网站建设风格
  • 装饰公司网站方案WordPress托管如果使用插件
  • 淄博网站排名公司网页设计与制作用什么软件做
  • 网站怎么才能被百度收录中国室内设计联盟网官网
  • 红色 网站西安市建设工程交易网
  • 展览展会网站建设商城站人工售票时间表
  • 个人网站备案查询桐乡网站设计
  • 网站买流量是怎么做的wordpress搭电影网站
  • 怎么用asp做网站修改wordpress图片外链
  • 服装网站建设环境分析网站横幅图片
  • 网站如何适应屏幕教学资源库 网站建设
  • 公司app与网站建设方案网站推广广告词大全集
  • 深圳建设网站费用万网代备案系统
  • 名片型网站开发wordpress添加媒体失败
  • 企业网站建设协议范本wordpress域名解析端口
  • 模板网站多钱Uie主题WordPress
  • 山东建站商城企业画册的设计效果
  • 证券公司网站建设方案百度域名怎么注册
  • 什么是开放式的网站个人做搜索网站违法吗
  • 做一个介绍网站多少钱快速免费建网站
  • 做简图的网站公司logo注册多少钱
  • 假冒建设银行网站博客用wordpress对吗
  • 公司网站建设济南兴田德润地址wordpress 相册模板
  • 青岛网站设计公司什么程序做网站收录好
  • 网站搜索引擎优化方案论文重庆梁平网站制作公司
  • 网站开发api平台北京竞价托管代运营
  • 企业网站网络推广关于网站建设的简历模板
  • 优易官方网站做包装一般看什么网站
  • wordpress的文章在哪做优化网站注意什么
  • 部门网站建设整改中国建设银行国际互联网站