c语言buffer用法_c++ stringbuffer

c语言buffer用法_c++ stringbufferringbuffer实质上是在一块连续的内存上对数据进行写入和读取,只是在写入和读取的时候有些不同。每次读取的时候都要从开辟空间的地址加上一个读取的偏移量读取,相对应的,写入的时候也要从开辟空间的地址加上一个写入的偏移量来写入。重点也就在读取和写入数据上面。下面从代码上面来说一下。首先定义一个ringbuffer的结构体typedefstruct{uint8_t*buf…

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

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

ring buffer实质上是在一块连续的内存上对数据进行写入和读取,只是在写入和读取的时候有些不同。每次读取的时候都要从开辟空间的地址加上一个读取的偏移量读取,相对应的,写入的时候也要从开辟空间的地址加上一个写入的偏移量来写入。重点也就在读取和写入数据上面。下面从代码上面来说一下。

  1. 首先定义一个ring buffer的结构体
typedef struct
{
    uint8_t *buffer;      //存放实际的数据
    int32_t read_offset;  //读取地址相对buffer的偏移量
    int32_t write_offset; //写入地址相对buffer的偏移量
    int32_t valid_size;   //buffer的有效size
    int32_t total_size;   //buffer的总大小,即init时malloc的size

} ring_buffer_t;
  1. 初始化ring buffer
/**
 1. 初始化ring_buffer
 2. malloc开辟传入的buff_size大小的空间存放buffer
 3. read_offset write_offset valid_size均置为0
*/
void ring_buffer_init(ring_buffer_t *ring_buffer, int32_t buff_size)
{
    ring_buffer->buffer = malloc(buff_size);
    memset(ring_buffer->buffer, 0, buff_size);

    ring_buffer->read_offset = 0;
    ring_buffer->write_offset = 0;
    ring_buffer->valid_size = 0;
    ring_buffer->total_size = buff_size;
}
  1. 释放ring buffer
/**
 *释放ring_buffer 
 */
void ring_buffer_deinit(ring_buffer_t *ring_buffer)
{
    if (ring_buffer->buffer != NULL)
    {
        free(ring_buffer->buffer);
    }
    memset(ring_buffer, 0, sizeof(ring_buffer_t));
}

4.向ring buffer里面写入数据,写入之后需要注意valid_size和write_offset的改变

/**
 * buffer_to_write:需要写入的数据的地址
 * size:需要写入的数据的大小
*/
void ring_buffer_write(void *buffer_to_write, int32_t size, ring_buffer_t *ring_buffer)
{
    int32_t write_offset = ring_buffer->write_offset;
    int32_t total_size = ring_buffer->total_size;
    int32_t first_write_size = 0;

    if (ring_buffer->valid_size + size > total_size) //ring_buffer->buffer未使用的总大小比需要写入的size小
    {
        printf("ring buffer is to short to save buff\n");
        printf("total_size:%d valid_size:%d size:%d\n", ring_buffer->total_size, ring_buffer->valid_size, size);
        return;
    }

    if (size + write_offset <= total_size) //ring_buffer->buffer的后段未写入的空间不小于size
    {
        memcpy(ring_buffer->buffer + write_offset, buffer_to_write, size);
    }
    else //ring_buffer->buffer的后段未写入的空间小于size,这时候需要先在后面写入一部分,然后返回头部,从前面接着写入
    {
        first_write_size = total_size - write_offset;
        memcpy(ring_buffer->buffer + write_offset, buffer_to_write, first_write_size);
        memcpy(ring_buffer->buffer, buffer_to_write + first_write_size, size - first_write_size);
    }
    ring_buffer->write_offset += size;
    ring_buffer->write_offset %= total_size;
    ring_buffer->valid_size += size;
}

附上图帮助理解:
在这里插入图片描述
5.从ring buffer里读取数据,写入之后需要注意valid_size和read_offset的改变

void ring_buffer_read(ring_buffer_t *ring_buffer, void *buff, int32_t size)
{
    int32_t read_offset = ring_buffer->read_offset;
    int32_t total_size = ring_buffer->total_size;
    int32_t first_read_size = 0;

    if (size > ring_buffer->valid_size)
    {
        printf("valid size < read size\n");
        printf("valid size:%d read size:%d\n", ring_buffer->valid_size, size);
        return;
    }

    if (total_size - read_offset >= size)
    {
        memcpy(buff, ring_buffer->buffer + read_offset, size);
    }
    else
    {
        first_read_size = total_size - read_offset;
        memcpy(buff, ring_buffer->buffer + read_offset, first_read_size);
        memcpy(buff + first_read_size, ring_buffer->buffer, size - first_read_size);
    }

    ring_buffer->read_offset += size;
    ring_buffer->read_offset %= total_size;
    ring_buffer->valid_size -= size;
}

希望本文能帮助到读者,最后附上一个测试:


int main()
{
    ring_buffer_t ring_buffer;
    uint8_t buf[8] = "1234abcd";
    uint8_t buf2[6] = {0};

    ring_buffer_init(&ring_buffer, RING_BUFFER_SIZE);//RING_BUFFER_SIZE我用宏定义为8

    ring_buffer_write(buf, sizeof(buf), &ring_buffer); //ring_buffer->buffer="1234abcd"

    ring_buffer_read(&ring_buffer, buf2, 6); //ring_buffer->buffer="1234abcd" buf2="1234ab"

    ring_buffer_write(buf + 2, 6, &ring_buffer); //ring_buffer->buffer="34abcdcd"

    ring_buffer_read(&ring_buffer, buf2, 6); //ring_buffer->buffer="34abcd78" buf2="7834ab"

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

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

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


相关推荐

  • BCGControlBar31 GUI Professional Crack

    BCGControlBar31 GUI Professional CrackBCGControlBarProforMFCversionsΩ578867473Version31.3.Released06/15/2021AnewclassCBCGPPropertyManager(seescreenshot)implementsaneasyandefficientwaytocreateyourapplicationproperties(options)fromXMLfile,bindpropertiestoCBCGPPr.

    2022年10月22日
    0
  • java开发后端框架_java开发后端框架[通俗易懂]

    java开发后端框架_java开发后端框架[通俗易懂]1java开发后端框架java从推出到现在技术不断发展,语言也优化的越来越好,对于java工程师来说技术的不断发展,他们需要不断学习java进阶,而对于新手来说就能从基础到核心。那么新手该怎么学习呢?下面小编给大家说说java开发后端框架,希望能对你有些帮助。Struts在项目中的作用Struts在项目主要起控制作用,只要用于web层(即视图层和控制层)Struts本身是使用典型的MVC结构实现…

    2022年5月1日
    50
  • [MAC] 编译安装和测试《魔兽世界》模拟服务端 TrinityCore

    2019独角兽企业重金招聘Python工程师标准>>>…

    2022年4月17日
    173
  • 深度揭秘强化学习技术与落地!智源大会「强化学习与决策智能」专题论坛

    深度揭秘强化学习技术与落地!智源大会「强化学习与决策智能」专题论坛决策智能是国家新一代人工智能的重要发展方向,强化学习是实现决策智能的核心技术之一。在强化学习中,智能体与环境进行不断的交互,基于环境的反馈学习如何选择一系列动作,以使长期累积的奖励和最大。…

    2022年6月2日
    97
  • 菜鸟实战UML——活动图

    菜鸟实战UML——活动图活动图活动图(activitydiagram)是阐明了业务用例实现的工作流程。业务工作流程说明了业务为向所服务的业务主角提供其所需要的价值而必须完成的工作,业务用例由一系列活动组成,他们共同为业务主角生成某些工件。工作流程通常包括一个基本的工作流程和一个或多个备选工作流程。工作流程的结构使用活动图来进行说明。用途活动图是UML用于对系统的动态行为建模的另一种常用工具,它描述活动的顺序,展现…

    2022年6月7日
    30
  • 网页视频下载(TS流下载合成)

    网页视频下载(TS流下载合成)前言最近《流浪地球》比较火,想找资源下载看看,无奈只找到了网址http://m.tlyy.tv/,但是我的chrome插件也嗅探不到网页上的视频。。于是乎,右击页面,inspect走起…步骤首先发现m3u8文件映入眼帘/偷笑,m3u8文件是什么文件呢,copyaddressandwget下来看看:文件playlist.m3u8内容如下,可见网页里的视频是根据这个play…

    2022年7月18日
    20

发表回复

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

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