ringbuffer的常规用法_likewise用法

ringbuffer的常规用法_likewise用法C/C++Linux服务器开发/后台架构师知识体系整理环形缓冲区(RINGBUFFER)的实现原理环形缓冲区通常有一个读指针和一个写指针(一个入指针和一个出指针)。读指针指向环形缓冲区中可读的数据,写指针指向环形缓冲区中可写的缓冲区。通过移动读指针和写指针就可以实现缓冲区的数据读取和写入。在通常情况下,环形缓冲区的读用户仅仅会影响读指针,而写用户仅仅会影响写指针。如果仅仅有一个读用户和一个写用户,那么不需要添加互斥保护机制就可以保证数据的正确性。如果有多个读写用户访问环形缓冲区,那么必须添加互斥保护机

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE使用 1年只要46元 售后保障 童叟无欺

C/C++Linux服务器开发/后台架构师知识体系整理

环形缓冲区(RING BUFFER)的实现原理

环形缓冲区通常有一个读指针和一个写指针(一个入指针和一个出指针)。读指针指向环形缓冲区中可读的数据,写指针指向环形缓冲区中可写的缓冲区。通过移动读指针和写指针就可以实现缓冲区的数据读取和写入。在通常情况下,环形缓冲区的读用户仅仅会影响读指针,而写用户仅仅会影响写指针。如果仅仅有一个读用户和一个写用户,那么不需要添加互斥保护机制就可以保证数据的正确性。如果有多个读写用户访问环形缓冲区,那么必须添加互斥保护机制来确保多个用户互斥访问环形缓冲区

环形缓冲区(RING BUFFER)实现原理图解

1、环形缓冲区(Ring Buffer)初始态
在这里插入图片描述
2、向环形缓冲区(Ring Buffer)中添加一个数据
在这里插入图片描述
3、向环形缓冲区(Ring Buffer)中添加一个数据,并读取一个数据
在这里插入图片描述
注意:环形缓冲区是使用的线性存储区实现的,实际的物理存储是线性的
类似于二维或多维数组,其实际存储也是由线下存储实现

RING BUFFER的用法(C语言)

话不多说直接上代码,边分析代码、边理解原理

RING BUFFER的常规用法

Ring Buffer的数据结构定义:

struct ring_buffer { 
   
    int in; /*写指针*/
    int out; /*读指针*/
    int size;  /*环形缓冲区大小*/
    char *buf; /*环形缓冲区*/
};

初始化Ring Buffer的接口:

bool RingBufferInit(struct ring_buffer *ring_buffer_p, int buf_size)
{ 
   
    ring_buffer_p->buf = (char *)malloc(buf_size);
    if (ring_buffer_p->buf) { 
   
        memset(fifo_p->buf, 0, buf_size);
    } else { 
   
        return false;
    }

    ring_buffer_p->size = buf_size;
    ring_buffer_p->head = 0;
    ring_buffer_p->tail = 0;
    return true;
}

判断Ring Buffer是否为空的接口:

bool RingBufferEmpty(struct ring_buffer *ring_buffer_p)
{ 
   
    return (ring_buffer_p->in == ring_buffer_p->out);
}

判断Ring Buffer是否为满的接口:

bool RingBufferFull(struct ring_buffer *ring_buffer_p)
{ 
   
    return (((ring_buffer_p->in) % ring_buffer_p->size) == ring_buffer_p->out);
}

读取Ring Buffer数据长度的接口:

int GetRingBufferLen(struct ring_buffer *ring_buffer_p)
{ 
   
    return (((ring_buffer_p>in - ring_buffer_p->out + ring_buffer_p->size) % ring_buffer_p->size);
}

清空Ring Buffer的接口:

void RingBufferClear(struct ring_buffer *ring_buffer_p)
{ 
   
    ring_buffer_p->in = 0;
    ring_buffer_p->out = 0;
}

释放Ring Buffer的接口:

void RingBufferClear(struct ring_buffer *ring_buffer_p)
{ 
   
    ring_buffer_p->in = 0;
    ring_buffer_p->out = 0;
    ring_buffer_p->size = 0;

    if(ring_buffer_p->buf) { 
   
        free(ring_buffer_p->buf);
        ring_buffer_p->buf = NULL;
    }
}

向Ring Buffer的写数据接口:

int RingBufferPut(struct ring_buffer *ring_buffer_p, char *buf, int len)
{ 
   
    int real_int_len = 0 , i = 0, surplus_buf_len = 0;

    if( len >= ring_buffer_p->size){ 
   
        return -1;
    }

    surplus_buf_len = ring_buffer_p->size - GetRingBufferLen(ring_buffer_p);
    if( len > surplus_buf_len )
        real_in_len = surplus_buf_len;
    } else if( len <= surplus_buf_len){ 
   
        real_in_len = len;
    }

    for( i = 0; i < real_in_len; i++){ 
   
        ring_buffer_p->buf[ ring_buffer_p->in % ring_buffer_p->size] = buf[i];
        ring_buffer_p->in = (ring_buffer_p->in + 1) % ring_buffer_p->size;
    }

    return i;
}

读取Ring Buffer数据的接口:

int RingBufferGet(struct ring_buffer *ring_buffer_p, char *buf, int buf_len)
{ 
   
    hd_s32_t real_out_len = 0, i =0, data_len = 0;

    data_len = GetRingBufferLen(ring_buffer_p);
    if( buf_len > data_len )
        real_out_len = data_len;
    } else if( buf_len <= data_len){ 
   
        real_out_len = buf_len;
    }

    for( i = 0; i < real_out_len; i++){ 
   
        buf[i] = ring_buffer_p->buf[ring_buffer_p->out % ring_buffer_p->size];
        ring_buffer_p->out = (ring_buffer_p->out + 1) % ring_buffer_p->size;
    }
    return i;
}

C/C++Linux服务器开发/后台架构师 系统性学习
C/C++Linux服务器开发/架构师面试题、学习资料、教学视频和学习路线图,免费分享有需要的可以自行添加学习交流群960994558

在这里插入图片描述

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/195189.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 怎么反编译dll文件(反编译dll查看源代码)

    DLL(DynamicLinkLibrary)文件为动态链接库文件,又称“应用程序拓展”,是软件文件类型。下载并解压.NETReflector,下面下载为绿色7.3.0.18版本;(注:此为英文版,百度有汉化版下载)链接:https://pan.baidu.com/s/1eoWFe62wA_uOdn3wTQ7e-w密码:gt87…

    2022年4月12日
    102
  • springmvc源码下载_web系统源码下载

    springmvc源码下载_web系统源码下载Spring源码下载、编译Spring源码下载Spring源码编译1、新增下载源地址2、修改依赖地址三级目录Spring源码下载注意:Spring源码使用的是Gradle,而不是Maven。因此下载Spring源码之前可以先安装Gradle,参考:Gradle的下载、安装和配置环境。Spring源码gitee地址:https://gitee.com/mirrors/Spring-Framework。Spring源码gitee仓库地址:https://gitee.com/mirrors/Spring

    2022年8月12日
    10
  • Dos攻击原理_防止xss攻击方法

    Dos攻击原理_防止xss攻击方法Technorati标签: DoS,攻击,网络防御,TCP,SYN_FloodTCP/IP协议的权限DoS(拒绝服务攻击)—–DenialofService该攻击的原理是利用TCP报文头来做的文章.下面是TCP数据段头格式。SourcePort和DestinationPort:是本地端口和目标端口SequenceNu

    2022年10月1日
    4
  • git从远程分支拉取代码_git更新分支下的代码

    git从远程分支拉取代码_git更新分支下的代码查看远程所有分支$gitbranch-a创建分支gitbranch[branchname]需要注意,创建分支时,不会有什么提示。新建分支并切换到指定分支gitcheckout-bmasterorigin/mastergitcheckout-b本地分支名origin/远程分支名拉取远程分支代码gitpullorigin/master使用上面的命令可以创建分支名并且关联到远程的分支,但是远程分支存在gitpush多次,发现关联分支的本地代码不是最新的,

    2026年2月6日
    2
  • windows如何在局域网下共享文件(传输文件、修改文件)

    windows如何在局域网下共享文件(传输文件、修改文件)前些天在了解Git版本控制的时候,看到了“局域网下可以共享文件(阅读or修改)”。发现自己之前都没了解过这个,虽然用着GitHub却对其来源的变化不甚了解。于是就动手操作了一下windows如何在局域网下共享文件。对,还有一些局域网下的传输软件。但我还没有去了解,所以在这里先不说了。什么是局域网局域网(LocalAreaNetwork,LAN),又称内网。指覆盖局部区域(如办公室…

    2022年5月4日
    209
  • 0x0000007e_c0000005改兼容性没用

    0x0000007e_c0000005改兼容性没用对于怎么解决应用程序正常初始化0xc0000005失败这个问题,小编觉得是需要知道的,因为我们在生活中遇到类似这样的问题几率还是蛮大的。所以小伙伴们要接着往下看哟~接下来小编就来告诉你们怎么解决应用程序正常初始化0xc0000005失败的问题。有的时候刷网页刷到一半,就突然间出现应用程序正常初始化0xc0000005失败的窗口提示,但是这是怎么回事呢?又该怎么解决呢?稳住,接下来小编就来告诉你们怎…

    2022年10月3日
    3

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号