无锁队列--基于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不仅是一个加 ...
随机推荐
- 面试中的Singleton
引子 “请写一个Singleton.”面试官微笑着和我说. “这可真简单.”我心里想着,并在白板上写下了下面的Singleton实现: 1 class Singleton 2 { 3 public ...
- Activity跳转时传递Bitmap对象的实现
前言 相信大家可能都了解Activity跳转时我们是能够传递參数的,比方使用Intent传递或者Bundle来传递,把当前Activity的一些信息传递给将要跳转到的新的Activity.可是不知道大 ...
- HDURevenge of Segment Tree(第二长的递增子序列)
HDURevenge of Segment Tree(第二长的递增子序列) 题目链接 题目大意:这题是求第二长的递增子序列. 解题思路:用n^2的算法来求LIS,可是这里还要记录一下最长的那个序列是否 ...
- [JBoss] JNDI与JBossNS
JNDI的作用 JNDI是 Java 命名与目录接口(Java Naming and Directory Interface). 随着分布式应用的发展,远程访问对象访问成为常用的方法.虽然说通过Soc ...
- android内存的一点优化
android手机给应用分配的内存通常是8兆左右,如果处理内存处理不当很容易造成OutOfMemoryError,我们的产品出现最多的错误也是OutOfMemoryError的异常, 在解决这个异常时 ...
- javascript每日一练(一)——javascript基础
一.javascript的组成 ECMAScript DOM BOM 二.变量类型 常见类型有:number, string, boolean, undefined, object, function ...
- Java返回类型泛型的用法小结
Java返回类型泛型的用法小结 版权声明:本文为博主原创文章,未经博主允许不得转载. 关于Java泛型的基本用法就不多说了,主要是一个编译期的检查,也避免了我们代码中的强制转换,比较经典的用法有泛型D ...
- Mac 安装配置启动Tomcat
Tomcat Mac 下的安装: TomCat 下载地址,例如: http://tomcat.apache.org/download-70.cgi 在Mac 上下载的时候,下载tar.gz包 下载完成 ...
- PHP移动互联网开发笔记(5)——文件的上传下载
原文地址:http://www.php100.com/html/php/rumen/2014/0326/6706.html 一.文件的上传 1.client设置: (1).在 标签中将enctype和 ...
- 报错消息写在AT SELECTION-SCREEN OUTPUT和START-OF-SELECTION事件下的区别
今天面试没答上来的问题,其实我是知道的,以前也遇到过.... START-OF-SELECTION下的话会在左下角报错 AT SELECTION-SCREEN OUTPUT消息会弹出框,然后点击就没有 ...