无锁队列--基于linuxkfifo实现
一直想写一个无锁队列,为了提高项目的背景效率。
有机会看到linux核心kfifo.h 原则。
所以这个实现自己仿照,眼下linux我们应该能够提供外部接口。
#ifndef _NO_LOCK_QUEUE_H_
#define _NO_LOCK_QUEUE_H_ #include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <string>
#include <pthread.h>
#include <iostream>
using namespace std; #ifndef max
#define max(x, y) ({ \
typeof(x) _max1 = (x); \
typeof(y) _max2 = (y); \
(void) (&_max1 == &_max2); \
_max1 > _max2 ? _max1 : _max2; })
#endif #ifndef min
#define min(x, y) ({ \
typeof(x) _min1 = (x); \
typeof(y) _min2 = (y); \
(void) (&_min1 == &_min2); \
_min1 < _min2 ? _min1 : _min2; })
#endif class Kfifo
{
public:
Kfifo(unsigned int isize);
~Kfifo();
unsigned int get(unsigned char *buffer, unsigned int len);
unsigned int put(const unsigned char *buffer, unsigned int len);
static unsigned long roundup_power_of_two(unsigned long val);
private:
inline bool is_power_of_2(unsigned long n)
{
return (n != 0 && ((n & (n - 1)) == 0));
};
inline unsigned int unused()
{
return (mask + 1) - (in - out);
} private:
unsigned int size;
unsigned int in;
unsigned int out;
unsigned int mask;
unsigned char* buffer;
}; struct proto
{
unsigned int msgid;
unsigned int cmd;
unsigned int info;
proto():msgid(0),cmd(0),info(0){}
}; #endif
实现文件
#include "MKfifo.h" Kfifo::~Kfifo()
{
if (buffer) free(buffer);
size = in = out=0;
}
unsigned long Kfifo::roundup_power_of_two(unsigned long val)
{
if (val & (val-1) == 0)
{
return val;
}
unsigned long maxulong = (unsigned long )((unsigned long ) ~0);
unsigned long andv = ~(maxulong&(maxulong>>1));
while((andv & val) == 0)
andv = andv>>1;
return andv<<1;
}
Kfifo::Kfifo(unsigned int isize):size(isize),in(0),out(0),mask(size - 1)
{
if (!is_power_of_2(isize))
{
size = roundup_power_of_two(isize);
}
buffer =(unsigned char*) malloc(isize);
} unsigned int Kfifo::get(unsigned char *_buffer, unsigned int len)
{
unsigned int l;
len = min(len, in - out);
__sync_synchronize(); l = min(len,size -(out&(size-1)));
memcpy(_buffer,buffer + (out& (size-1)),l);
memcpy(_buffer + l,buffer,len - l); __sync_synchronize(); out +=len;
return len; }
unsigned int Kfifo::put(const unsigned char *_buffer, unsigned int len)
{
unsigned int l;
len = min(len, size - in + out);
__sync_synchronize();
l = min(len, size - (in & (size - 1)));
memcpy(buffer + (in & (size - 1)), _buffer, l);
memcpy(buffer, _buffer + l, len - l);
__sync_synchronize();
in += len;
return len; } void * consumer(void * arg)
{
printf("consumer\n");
Kfifo* fifo = (Kfifo*) arg;
if (!fifo)
{
return NULL;
} for (;;)
{
proto p;
unsigned int len = fifo->get((unsigned char*)&p,sizeof(p));
if (len>0)
{
cout << "~~~~~~~~~~~~~~~~~~~~"<<endl;
cout << "consumer proto msg id :"<<p.msgid<<endl;
cout << "consumer proto msg cmd :"<<p.cmd<<endl;
cout << "consumer proto msg info :"<<p.info<<endl;
cout << "~~~~~~~~~~~~~~~~~~~~"<<endl;
}
} return (void *)fifo;
}
void* producer(void* args)
{
Kfifo* fifo = (Kfifo*) args; if (!fifo)
{
return NULL;
}
unsigned int i=0;
for (;;)
{
proto p;
p.msgid = i++;
p.cmd = 333;
p.info = 44444;
fifo->put((const unsigned char*)&p,sizeof(p));
cout<<"producer put msgid :"<<p.msgid<<endl;
}
return (void*)fifo;
}
int main()
{
Kfifo *fifo = new Kfifo(1024);
pthread_t consumeid,producerid;
pthread_create(&producerid,NULL,producer,(void*)fifo);
pthread_create(&consumeid,NULL,consumer,(void*)fifo); printf("info!!\n");
pthread_join(consumeid,NULL);
pthread_join(producerid,NULL);
return 0;
}
可经过测试。我将基于队列。其丰富的应用。
无锁队列--基于linuxkfifo实现的更多相关文章
- 基于无锁队列和c++11的高性能线程池
基于无锁队列和c++11的高性能线程池线程使用c++11库和线程池之间的消息通讯使用一个简单的无锁消息队列适用于linux平台,gcc 4.6以上 标签: <无> 代码片段(6)[ ...
- 基于folly的AtomicIntrusiveLinkedList无锁队列进行简单封装的多生产多消费模型
1.基于folly的AtomicIntrusiveLinkedList略微修改的无锁队列代码: #ifndef FOLLY_REVISE_H #define FOLLY_REVISE_H namesp ...
- boost 无锁队列
一哥们翻译的boost的无锁队列的官方文档 原文地址:http://blog.csdn.net/great3779/article/details/8765103 Boost_1_53_0终于迎来了久 ...
- 一个可无限伸缩且无ABA问题的无锁队列
关于无锁队列,详细的介绍请参考陈硕先生的<无锁队列的实现>一文.然进一步,如何实现一个不限node数目即能够无限伸缩的无锁队列,即是本文的要旨. 无锁队列有两种实现形式,分别是数组与链表. ...
- folly无锁队列,尝试添加新的函数(续)
基于上一篇文章,dropHead取出节点后,删除节点,会出现内存访问的问题.按照这个逻辑,如果将移出的节点保存到一个无锁队列中,然后在需要节点的时候,从这个备用的无锁队列中取出节点,那么应该就可以避开 ...
- DIOCP开源项目-高效稳定的服务端解决方案(DIOCP + 无锁队列 + ZeroMQ + QWorkers) 出炉了
[概述] 自从上次发布了[DIOCP开源项目-利用队列+0MQ+多进程逻辑处理,搭建稳定,高效,分布式的服务端]文章后,得到了很多朋友的支持和肯定.这加大了我的开发动力,经过几个晚上的熬夜,终于在昨天 ...
- 高性能无锁队列 Disruptor 初体验
原文地址: haifeiWu和他朋友们的博客 博客地址:www.hchstudio.cn 欢迎转载,转载请注明作者及出处,谢谢! 最近一直在研究队列的一些问题,今天楼主要分享一个高性能的队列 Disr ...
- 锁、CAS操作和无锁队列的实现
https://blog.csdn.net/yishizuofei/article/details/78353722 锁的机制 锁和人很像,有的人乐观,总会想到好的一方面,所以只要越努力,就会越幸运: ...
- 【DPDK】【ring】从DPDK的ring来看无锁队列的实现
[前言] 队列是众多数据结构中最常见的一种之一.曾经有人和我说过这么一句话,叫做“程序等于数据结构+算法”.因此在设计模块.写代码时,队列常常作为一个很常见的结构出现在模块设计中.DPDK不仅是一个加 ...
随机推荐
- 解决yum升级的问题“There was a problem importing one of the Python modules”
yum命令升级的时候,报出这个错误. There was a problem importing one of the Python modules required to run yum. The ...
- Win32 Windows编程 十二
一.对话框 1.对话框的分类 2.对话框的基本使用方式 3.对话框资源 4.有模式对话框的使用 5. 无模式对话框的使用 5.1 加入对话框资源 5.2 定义窗体处理函数 BOOL CALLBACK ...
- Bootstrap "row"类宽度超过问题
问题原因: VOORBootstrap门格系统布局,类别col-xs-*身边有15px的padding,在这样的元素img我们希望展现的顶部边缘,这需要col-xs-*式:padding:0px; 如 ...
- longest incresing sequence
动态规划基本题目,longest incresing sequence,找出序列中的最长递增子序列: 例如给出序列{8,3,5,2,4,9,7,11}, 其中最长递增子序列为{3,5,9,11}或{3 ...
- VC Office2007界面对话框实现
我们知道VS2008SP1之后,MFC就多了一个功能包,可以快速的建立一个ribbon的界面,视觉样式可以在office 2007蓝.黑等颜色之间切换,这对于单文档/多文档做界面非常方便,而且也蛮好看 ...
- 最近盯着accesslog看,发现许多奇怪的东东
1.spider,各式各样的spider,就像海里的游鱼 有大的,有小的 2.各类探测http代理的spider,比如这种日志 60.173.14.85 - - [03/Sep/2013:09:59: ...
- 09-使用for循环输出空心菱形(循环)
/** * 使用for循环输出空心菱形 * */ public class Test7 { public static void main(String[] args) { for (int i = ...
- Winform - TreeView控件,只展开根目录
TreeNode类型是有Expand和ExpandAll这两个方法.而Treeview是只有ExpandAll的,想要展开根目录下面的节点的话 //只展开根目录 ) ].Expand();
- 熬之滴水成石:最想深入了解的内容--windows内核机制(15)
66--内存管理(4) 说说在windows中内存空间初始化的事,开始的开始通过处理器的分页机制,预先建立相应足够的页表以便页表来访问物理内存.预先建立的这个物理内存的是windows自己的加载程序, ...
- 摘记:IIS长时间任务超时处理
1.如果是在客户端调用Webservice,首先设置客户端超时, SoapHttpClientProtocol.Timeout = 3600 * 1000;//单位为秒,这是设置了一个小时 2.如果站 ...