什么是可epoll队列?

就可以使用epoll来监控队列中是否有数据的队列,当然也支持select和poll。

应用场景

一个线程,需要将队列(共享内存队列或普通队列均可)中的数据取出来,然后通过网络发送出去。如果没有可epoll队列,这个问题处理起来就比较麻烦。

代码实现

实现基于pipe,但pipe可能会产生毛刺。新的内存(2.6.22)引入了eventfd(相关的还有timerfd和signalfd),基于它的实现,不会有毛刺。

/** 可以放入Epoll监控的队列

* RawQueueClass为原始队列类名,如util::CArrayQueue

* 为线程安全类

*/

template <class RawQueueClass>

class CEpollableQueue: public CEpollable

{

typedef typename RawQueueClass::_DataType DataType;

public:

/** 构造一个可Epoll的队列,注意只可监控读事件,也就是队列中是否有数据

* @queue_max: 队列最大可容纳的元素个数

* @exception: 如果出错,则抛出CSyscallException异常

*/

CEpollableQueue(uint32_t queue_max)

:_raw_queue(queue_max)

,_push_waiter_number(0)

{

if (-1 == pipe(_pipefd)) throw sys::CSyscallException(errno, __FILE__, __LINE__);

set_fd(_pipefd[0]);

}

~CEpollableQueue()

{

close();

}

/** 关闭队列 */

virtual void close()

{

sys::LockHelper<sys::CLock> lock_helper(_lock);

if (_pipefd[0] != -1)

{

// 让CEpollable来关闭_pipefd[0],在CEpollable::close()中将会调用

// before_close,以保持语义总是相同的

CEpollable::close();

//close_fd(_pipefd[0]);

_pipefd[0] = -1;

}

if (_pipefd[1] != -1)

{

close_fd(_pipefd[1]);

_pipefd[1] = -1;

}

}

/** 判断队列是否已满 */

bool is_full() const

{

sys::LockHelper<sys::CLock> lock_helper(_lock);

return _raw_queue.is_full();

}

/** 判断队列是否为空 */

bool is_empty() const

{

sys::LockHelper<sys::CLock> lock_helper(_lock);

return _raw_queue.is_empty();

}

/***

* 取队首元素

* @elem: 存储取到的队首元素

* @return: 如果队列为空,则返回false,否则返回true

*/

bool front(DataType& elem) const

{

sys::LockHelper<sys::CLock> lock_helper(_lock);

if (_raw_queue.is_empty()) return false;

elem = _raw_queue.front();

return true;

}

/***

* 弹出队首元素

* @elem: 存储弹出的队首元素

* @return: 如果队列为空,则返回false,否则取到元素并返回true

* @exception: 如果出错,则抛出CSyscallException异常

*/

bool pop_front(DataType& elem)

{

sys::LockHelper<sys::CLock> lock_helper(_lock);

return do_pop_front(elem);

}

void pop_front()

{

DataType elem;

(void)pop_front(elem);

}

/***

* 从队首依次弹出多个元素

* @elem_array: 存储弹出的队首元素数组

* @array_size: 输入和输出参数,存储实际弹出的元素个数

* @exception: 如果出错,则抛出CSyscallException异常

*/

void pop_front(DataType* elem_array, uint32_t& array_size)

{

uint32_t i = 0;

sys::LockHelper<sys::CLock> lock_helper(_lock);

for (;;)

{

if (!do_pop_front(elem_array[i])) break;

if (++i == array_size) break;

}

array_size = i;

}

/***

* 向队尾插入一元素

* @elem: 待插入到队尾的元素

* @millisecond: 如果队列满,等待队列非满的毫秒数,如果为0则不等待,直接返回false

* @return: 如果队列已经满,则返回false,否则插入成功并返回true

* @exception: 如果出错,则抛出CSyscallException异常

*/

bool push_back(DataType elem, uint32_t millisecond=0)

{

sys::LockHelper<sys::CLock> lock_helper(_lock);

while (_raw_queue.is_full())

{

// 立即返回

if (0 == millisecond) return false;

// 超时等待

util::CountHelper<volatile int32_t> ch(_push_waiter_number);

if (!_event.timed_wait(_lock, millisecond))

{

return false;

}

}

char c = 'x';

_raw_queue.push_back(elem);

// write还有相当于signal的作用

while (-1 == write(_pipefd[1], &c, sizeof(c)))

{

if (errno != EINTR)

throw sys::CSyscallException(errno, __FILE__, __LINE__);

}

return true;

}

/** 得到队列中当前存储的元素个数 */

uint32_t size() const

{

sys::LockHelper<sys::CLock> lock_helper(_lock);

return _raw_queue.size();

}

private:

bool do_pop_front(DataType& elem)

{

// 没有数据,也不阻塞,如果需要阻塞,应当使用事件队列CEventQueue

if (_raw_queue.is_empty()) return false;

char c;

// read还有相当于CEvent::wait的作用

while (-1 == read(_pipefd[0], &c, sizeof(c)))

{

if (errno != EINTR)

throw sys::CSyscallException(errno, __FILE__, __LINE__);

}

elem = _raw_queue.pop_front();

// 如果有等待着,则唤醒其中一个

if (_push_waiter_number > 0) _event.signal();

return true;

}

private:

int _pipefd[2]; /** 管道句柄 */

sys::CEvent _event;

mutable sys::CLock _lock;

RawQueueClass _raw_queue; /** 普通队列实例 */

volatile int32_t _push_waiter_number; /** 等待队列非满的线程个数 */

};

可epoll队列的更多相关文章

  1. (转载) Linux IO模式及 select、poll、epoll详解

    注:本文是对众多博客的学习和总结,可能存在理解错误.请带着怀疑的眼光,同时如果有错误希望能指出. 同步IO和异步IO,阻塞IO和非阻塞IO分别是什么,到底有什么区别?不同的人在不同的上下文下给出的答案 ...

  2. Linux epoll

    一. epoll函数集 epoll主要有三个函数: 1. int epoll_create(int size); 创建一个epoll的句柄,size用来告诉内核这个监听的数目一共有多大.这个参数不同于 ...

  3. IO多路复用之epoll总结

    1.基本知识 epoll是在2.6内核中提出的,是之前的select和poll的增强版本.相对于select和poll来说,epoll更加灵活,没有描述符限制.epoll使用一个文件描述符管理多个描述 ...

  4. Linux Epoll相关知识

    其实在Linux下设计并发网络程序,向来不缺少方法,比如典型的Apache模型(Process Per Connection,简称PPC),TPC(Thread PerConnection)模型,以及 ...

  5. 基本I/O模型与Epoll简介

    5种基本的I/O模型:1)阻塞I/O ;2)非阻塞I/O; 3)I/O复用(select和poll);4)信号驱动I/O(SIGIO);5)异步I/O(POSIX.1的aio_系列函数). 操作系统中 ...

  6. Linux epoll 笔记(高并发事件处理机制)

    wiki: Epoll优点: Epoll工作流程: Epoll实现机制: epollevent; Epoll源码分析: Epoll接口: epoll_create; epoll_ctl; epoll_ ...

  7. UNIX网络编程学习指南--epoll函数

    epoll是select/poll的强化版,都是多路复用的函数,epoll有了很大的改进. epoll的功能 1.支持监听大数目的socket描述符 一个进程内,select能打开的fd是有限制的,有 ...

  8. 基本套接字编程(5) -- epoll篇

    1. epoll技术 epoll是Linux内核为处理大批量文件描述符而作了改进的poll,是Linux下多路复用IO接口select/poll的增强版本,它能显著提高程序在大量并发连接中只有少量活跃 ...

  9. Server Develop (六) Linux epoll总结

    Linux  epoll epoll是Kernel 2.6后新加入的事件机制,在高并发条件下,远优于select.epoll最大的好处在于它不会随着监听fd数目的增长而降低效率.因为在内核中的sele ...

随机推荐

  1. 【BZOJ】2809: [Apio2012]dispatching(左偏树)

    题目 传送门:QWQ 分析 显然是一个资瓷合并的堆 现学了一发左偏树:教程 然后就没了 代码 #include <bits/stdc++.h> #define lc son[x][0] # ...

  2. *(ptr++) += 123

    *(ptr++) += 123; 等价于:*(ptr) = *(ptr) + 123; ptr++; 而不是:*(ptr++) = *(ptr++) + 123;程序员面试宝典p32 #include ...

  3. JavaScript中的可枚举属性与不可枚举属性

    在JavaScript中,对象的属性分为可枚举和不可枚举之分,它们是由属性的enumerable值决定的.可枚举性决定了这个属性能否被for…in查找遍历到. 一.怎么判断属性是否可枚举 js中基本包 ...

  4. Spring中引质增强的安全

    在引质增强中使用ThreadLocal变量,是因为控制状态使代理类变成了非线程安全的实例,为了解决单线程安全的问题,通过ThreadLocal让每个线程单独使用一个状态.

  5. 第二章:Android Studio概述(一)[学习Android Studio汉化教程]

     Android Studio是一个视窗化的开发环境.为了充分利用有限的屏幕空间,不让你束手束脚,Android Studio 在特定的时间仅仅显示一小部分可用窗口. 除了一些上下文敏感的窗口和上下文 ...

  6. JQueryDOM节点操作

    你一.JQueryDom节点操作 2.1查找节点 获取p节点 var $pDm=$("p"); 输出p节点的title属性 alert($pDm.attr("title& ...

  7. HTTP接口开发专题二(发送http请求的接口工具类)

    import java.io.BufferedReader; import java.io.ByteArrayOutputStream; import java.io.IOException; imp ...

  8. IOS 阻止 锁屏

    [UIApplication sharedApplication].idleTimerDisabled=YES;不自动锁屏 idleTimerDisabled

  9. Java 单例模式详解(转)

    概念: java中单例模式是一种常见的设计模式,单例模式分三种:懒汉式单例.饿汉式单例.登记式单例三种. 单例模式有一下特点: 1.单例类只能有一个实例. 2.单例类必须自己自己创建自己的唯一实例. ...

  10. SPI protocol驱动编写实例

    内核版本:3.9.5 Linux中SPI驱动有俩个部分组成:controller驱动,直接和底层硬件打交道,protocol驱动,针对特定的设备,也是我们要做的. 这里只考虑SPI protocol驱 ...