C 语言中实现环形缓冲区_C 语言

1.实现代码:

#include
#include
#include
#include
#include 

#define BUFFSIZE 1024 * 1024
#define min(x, y) ((x) < (y) ? (x) : (y)) 

pthread_mutex_t lock = PTHREAD_MUTEX_INITIALIZER; 

struct cycle_buffer {
 unsigned char *buf;
 unsigned int size;
 unsigned int in;
 unsigned int out;
 pthread_mutex_t lock;
}; 

static struct cycle_buffer *fifo = NULL; 

static int init_cycle_buffer(void)
{
 int size = BUFFSIZE, ret; 

 ret = size & (size - 1);
 if (ret)
  return ret;
 fifo = (struct cycle_buffer *) malloc(sizeof(struct cycle_buffer));
 if (!fifo)
  return -1; 

 memset(fifo, 0, sizeof(struct cycle_buffer));
 fifo->size = size;
 fifo->in = fifo->out = 0;
 pthread_mutex_init(&fifo->lock, NULL);
 fifo->buf = (unsigned char *) malloc(size);
 if (!fifo->buf)
  free(fifo);
 else
  memset(fifo->buf, 0, size);
 return 0;
} 

unsigned int fifo_get(unsigned char *buf, unsigned int len)
{
 unsigned int l;
 len = min(len, fifo->in - fifo->out);
 l = min(len, fifo->size - (fifo->out & (fifo->size - 1)));
 memcpy(buf, fifo->buf + (fifo->out & (fifo->size - 1)), l);
 memcpy(buf + l, fifo->buf, len - l);
 fifo->out += len;
 return len;
} 

unsigned int fifo_put(unsigned char *buf, unsigned int len)
{
 unsigned int l;
 len = min(len, fifo->size - fifo->in + fifo->out);
 l = min(len, fifo->size - (fifo->in & (fifo->size - 1)));
 memcpy(fifo->buf + (fifo->in & (fifo->size - 1)), buf, l);
 memcpy(fifo->buf, buf + l, len - l);
 fifo->in += len;
 return len;
} 

static void * thread_read(void *arg)
{
 char buf[1024];
 unsigned int n;
 pthread_detach(pthread_self());
 for (;;) {
  memset(buf, 0, sizeof(buf));
  pthread_mutex_lock(&fifo->lock);
  n = fifo_get(buf, sizeof(buf));
  pthread_mutex_unlock(&fifo->lock);
  write(STDOUT_FILENO, buf, n);
 }
 printf("nnafter thread_read : %snn",buf);
 return NULL;
} 

static void * thread_write(void *arg)
{
 unsigned char buf[] = "hello world";
 pthread_detach(pthread_self());
 for (;;) {
  pthread_mutex_lock(&fifo->lock);
  fifo_put(buf, strlen(buf));
  pthread_mutex_unlock(&fifo->lock);
 }
 return NULL;
} 

int main(void)
{
 int ret;
 pthread_t wtid, rtid;
 ret = init_cycle_buffer();
 if (ret == -1)
  return ret; 

 pthread_create(&wtid, NULL, thread_write, NULL);
 pthread_create(&rtid, NULL, thread_read, NULL);
 pthread_exit(NULL);
 return 0;
}

1.buffer指向存放数据的缓冲区,size是缓冲区的大小,in是写指针下标,out是读指针下标,在len和(fifo->size - fifo->in + fifo->out)之间取一个较小的值赋给len。注意,当(fifo->in == fifo->out+fifo->size)时,表示缓冲区已满,此时得到的较小值一定是0,后面实际写入的字节数也全为0。另一种边界情况是当len很大时(因为len是无符号的,负数对它来说也是一个很大的正数),这一句也能保证len取到一个较小的值,因为fifo->in总是大于等于fifo->out,所以后面的那个表达式的值不会超过fifo->size的大小把上一步决定的要写入的字节数len“切开”,这里又使用了一个技巧。注意:实际分配给fifo->buffer的字节数fifo->size,必须是2的幂,否则这里就会出错。既然fifo->size是2的幂,那么 (fifo->size-1)也就是一个后面几位全为1的数,也就能保证(fifo->in & (fifo->size - 1))总为不超过(fifo->size - 1)的那一部分,和(fifo->in)% (fifo->size - 1)的效果一样。 

 2.这样后面的代码就不难理解了,它先向fifo->in到缓冲区末端这一块写数据,如果还没写完,在从缓冲区头开始写入剩下的,从而实现了循环缓冲。最后,把写指针后移len个字节,并返回len。

 3.从上面可以看出,fifo->in的值可以从0变化到超过fifo->size的数值,fifo->out也如此,但它们的差不会超过fifo->size 。

 以上就是环形缓冲区域的C语言实现详解,希望对大家有所帮助,谢谢支持!

以上是小编为您精心准备的的内容,在的博客、问答、公众号、人物、课程等栏目也有的相关内容,欢迎继续使用右上角搜索按钮进行搜索c
环形缓冲区
c语言实现环形缓冲区、环形缓冲区 c语言、c语言实现循环缓冲区、环形缓冲区的实现原理、java实现环形缓冲区,以便于您获取更多的相关知识。

时间: 2024-07-29 23:16:39

C 语言中实现环形缓冲区_C 语言的相关文章

详细解析C语言中的开方实现_C 语言

关于C语言中的开方计算,首先想到的当然是sqrt()函数,让我们先来回顾一下它的基本用法: 头文件:#include <math.h> sqrt() 用来求给定值的平方根,其原型为: double sqrt(double x); 参数 x 为要计算平方根的值. 如果 x < 0,将会导致 domain error 错误,并把全局变量 errno 的值为设置为 EDOM. 返回值 返回 x 平方根. 注意,使用 GCC 编译时请加入-lm. 实例计算200 的平方根值. #include

深入浅析C语言中堆栈和队列_C 语言

1.堆和栈 (1)数据结构的堆和栈 堆栈是两种数据结构. 栈(栈像装数据的桶或箱子):是一种具有后进先出性质的数据结构,也就是说后存放的先取,先存放的后取.这就如同要取出放在箱子里面底下的东西(放入的比较早的物体),首先要移开压在它上面的物体(放入的比较晚的物体). 堆(堆像一棵倒过来的树):是一种经过排序的树形数据结构,每个结点都有一个值.通常所说的堆的数据结构,是指二叉堆.堆的特点是根结点的值最小(或最大),且根结点的两个子树也是一个堆.由于堆的这个特性,常用来实现优先队列,堆的存取是随意,

模拟实现C语言中的内存管理_C 语言

这里模拟了C语言中的内存管理,当我们要创建或者使用一个对象时,那么这个对象会调用retain方法,计数+1,当我们要释放对象,我们会调用free,这里注意要对计数记性判断,如果是0的话,那么就会销毁. #import <Foundation/Foundation.h> int cnt = 0; void fun (charchar * p) { printf("%c\n",p[0]); } charchar * retain1(charchar * p) { //retai

简单总结C语言中的运算符优先级_C 语言

C语言中有很多运算符,除了四则运算,还有位运算.比较运算.逻辑运算.赋值运算等等,令人眼花缭乱的同时,优先级也让人头疼.通常建议在写代码的时候不用省略括号,但是并不是所有程序员都会按照这个规矩来,因此还是有必要记录一下,以备查阅. 总结几个比较重要的原则: 取成员的"0"目运算符[() . -> []]优先级最高: 单目运算符高于双目运算符: 四则运算高于移位运算: 移位运算高于比较运算: 比较运算高于位运算,位运算高于逻辑运算: 逻辑运算高于赋值运算: 逗号永远最低.

浅析C语言中的内存布局_C 语言

本节注重分清几个概念:.text .data .bss   堆   栈    静态存储区    只读存储区等 从程序到a.out 把程序变成.text  .data  .bss  是编译原理完成的过程 从a.out把程序映射到对应的内存地址空间是操作系统完成的,也就是在操作系统创建进程的时候完成的,在描述进程的那个结构体中. 我们常说的堆是为了申请动态内存的时候使用的,malloc. 栈是为了在函数中切换使用的,即存放函数中的局部变量.(堆和栈是操作系统分配的,所有不在a.out中) 静态存储区

深入解析C语言中常数的数据类型_C 语言

废话不多说,上代码 复制代码 代码如下: //编译环境:codeblocks+gcc#include <stdio.h>#include <stdint.h>int Fun(){    uint64_t y;    uint32_t x1, x2;     //y = 3000 * 24000000 / 1000;//常数默认作为32位数据,临时运算结果也是32位,溢出错误    //y = (uint64_t)3000 * (uint64_t)24000000 / 1000;//

Go语言中关闭带缓冲区的频道实例分析_Golang

本文实例分析了Go语言中关闭带缓冲区的频道.分享给大家供大家参考.具体分析如下: Go语言提供了两种频道,带缓冲区和不带缓冲区的.不带缓冲区的频道,发送和接收是同步的,必须接收端接收了消息,发送端才能从发送调用中解脱.带缓冲区的频道,在缓冲区满之前,发送和接收是异步的,发送端的发送操作只保证把消息放入缓冲区. Go的频道是可以关闭的,关闭频道的目的是让接收端知道不会再有消息从这个频道进入,我们可能会用某个频道的关闭来表示某种状态的终结. 当我们关闭一个带缓冲区的频道时,如果缓冲区中还有消息,接收

C语言数据类型转换实例代码_C 语言

数据类型转换就是将数据(变量.表达式的结果)从一种类型转换到另一种类型.例如,为了保存小数你可以将int类型的变量转换为double类型. 数据类型转换的一般格式为: (type_name) expression type_name为要转换到的数据类型,expression为表达式.例如: (float) a; //把a转换为实型 (int)(x+y); //把x+y的结果转换为整型 (float) 100; //将一个常量转换为实型 [示例]将整数转换为浮点数: #include <stdio

VC实现A进程窗口嵌入到B进程窗口中显示的方法_C 语言

本文通过一个Demo示例讲述把A应用程序嵌入到B应用程序中显示的方法. 主要代码如下: //在B应用启动时创建A进程 CreateProcess(_T("A.exe"),NULL,NULL,NULL,FALSE,CREATE_NEW_CONSOLE,NULL,NULL,NULL,NULL); Sleep(30); HWND hWndChild = FindWindow(_T("AAA"),_T("AAA")); while(!hWndChild)