无锁队列--基于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不仅是一个加 ...
随机推荐
- 查询SystemFeature的方法
查询SystemFeature的方法可以在adb shell下敲如下的命令: dumpsys package 然后搜feature关键字. 例如,我的平台的SystemFeature,如下所示: Fe ...
- iOS经常使用加密方式(MD5,AES,BASE64)与网络数据安全
演示样例项目下载地址 https://github.com/cerastes/Encryption 1MD5 创建MD5类 #import <Foundation/Foundation.h&g ...
- 设计模式(Abstract Factory)抽象工厂
1. 需求: 设计一个电脑组装程序,对于组装品牌电脑. 用零件组装(主板.硬盘.显示器)由品牌提供的所有. 让我们组装一台联想电脑,板子.由联想提供. (眼下仅仅有Lenovo和Dell两种品牌) 2 ...
- Linux安装完Tomcat后无法登陆管理界面
今天在Linux中安装完Tomcat后无法登陆Tomcat的管理界面,也就无法利用Tomcat管理界面来部署项目. 在Windows中一般配置完Tomcat后,只要在[conf]目录下的“tomcat ...
- Android 讲述Help提示框
Android 讲述Help提示框 XML/HTML代码 <stringname="help_dialog_text"> <i>Author:fonter. ...
- 【剑指offer】Q38:数字在数组中出现的次数
与折半查找是同一个模式,不同的是,在这里不在查找某个确定的值,而是查找确定值所在的上下边界. def getBounder(data, k, start, end, low_bound = False ...
- 最短路径A*算法原理及java代码实现(看不懂是我的失败)
算法仅仅要懂原理了,代码都是小问题,先看以下理论,尤其是红色标注的(要源代码请留下邮箱,有測试用例,直接执行就可以) A*算法 百度上的解释: A*[1](A-Star)算法是一种静态路网中求解最短路 ...
- (摘录)26个ASP.NET常用性能优化方法
数据库访问性能优化 数据库的连接和关闭 访问数据库资源需要创建连接.打开连接和关闭连接几个操作.这些过程需要多次与数据库交换信息以通过身份验证,比较耗费服务器资源. ASP.NET中提供了连接池(Co ...
- UC网盘被关停 将与阿里淘盘合并?(案例说明云盘的成本才是重点,技术不是问题;与巨头竞争是重点)
UC网盘被关停 将与阿里淘盘合并? 二八2016-03-18 08:05:36阿里巴巴 网盘 产品阅读(0)评论(12) UC网盘项目组日前发布公告称将停止网盘存储服务,原因为配合国家对云盘传播淫秽色 ...
- HTTP协议和web工作原理
本章学完之后能干什么? 要把 知识点学好,那就需要把它相关的周边知识点了解全面 HTTP协议是web学习的核心!!! 学东东切忌只学配置,不学原理:只学会框架有什么用,要会自己写框架!! web学习直 ...