按照lighttpd的方式封装了一下epoll,打算以后就直接这么用了,虽然简陋了点,不过很容易修改
event.h头文件
#include <sys/epoll.h>
#define BV(x) (1 << x)
#define FDEVENT_IN BV(0)
#define FDEVENT_PRI BV(1)
#define FDEVENT_OUT BV(2)
#define FDEVENT_ERR BV(3)
#define FDEVENT_HUP BV(4)
#define FDEVENT_NVAL BV(5)
typedef void (*fdevent_handler)(int fd,void *ctx, int revents);
typedef struct _fdnode{
fdevent_handler handler;
int fd;
void *ctx;
int status;
}fdnode;
typedef struct fdevents{
fdnode **fdarray;
size_t maxfds;
int epoll_fd;
struct epoll_event *epoll_events;
}fdevents;
fdevents *fdevent_init(size_t maxfds);
void fdevent_free(fdevents *ev);
int fdevent_register(fdevents *ev,int fd,fdevent_handler handler,void *ctx);
int fdevent_unregister(fdevents *ev,int fd);
int fdevent_event_add(fdevents *ev,int fd,int events);
int fdevent_event_del(fdevents *ev,int fd);
int fdevent_poll(fdevents *ev, int timeout_ms);
event.c实现
#include<stdio.h>
#include<unistd.h>
#include<sys/types.h>
#include<stdlib.h>
#include<sys/socket.h>
#include<errno.h>
#include<fcntl.h>
#include<netinet/in.h>
#include<sys/resource.h>
#include "event.h"
//foward declaration
static fdnode *fdnode_init();
static void fdnode_free(fdnode *fdn);
fdevents *fdevent_init(size_t maxfds){
fdevents *ev;
ev = calloc(1,sizeof(*ev));
ev->fdarray = calloc(maxfds , sizeof(*ev->fdarray));
ev->maxfds = maxfds;
ev->epoll_fd=epoll_create(maxfds);
if(-1==ev->epoll_fd){
fprintf(stderr,"epoll create failed,%s\n",strerror(errno));
return NULL;
}
ev->epoll_events=malloc(maxfds * sizeof(*ev->epoll_events));
return ev;
}
void fdevent_free(fdevents *ev){
size_t i;
if(!ev) return ;
close(ev->epoll_fd);
free(ev->epoll_events);
for(i=0;i<ev->maxfds;i++){
if(ev->fdarray[i]) free(ev->fdarray[i]);
}
free(ev->fdarray);
free(ev);
}
int fdevent_register(fdevents *ev,int fd,fdevent_handler handler,void *ctx){
fdnode *fdn;
fdn=fdnode_init();
fdn->handler=handler;
fdn->fd=fd;
fdn->ctx=ctx;
ev->fdarray[fd]=fdn;
return 0;
}
int fdevent_unregister(fdevents *ev,int fd){
if(!ev) return 0;
fdnode *fdn=ev->fdarray[fd];
fdnode_free(fdn);
ev->fdarray[fd]=NULL;
return 0;
}
int fdevent_event_add(fdevents *ev,int fd,int events){
struct epoll_event ep;
int add=0;
fdnode *fdn=ev->fdarray[fd];
add = (fdn->status == -1 ? 1 : 0);
memset(&ep,0,sizeof(ep));
ep.events = 0;
if(events & FDEVENT_IN) ep.events |=EPOLLIN;
if(events & FDEVENT_OUT) ep.events |=EPOLLOUT;
ep.data.ptr=NULL;
ep.data.fd=fd;
if(0 != epoll_ctl(ev->epoll_fd,add ? EPOLL_CTL_ADD : EPOLL_CTL_MOD,fd,&ep)){
fprintf(stderr,"epoll_ctl failed %s\n",strerror(errno));
return -1;
}
if(add){
fdn->status=1;
}
return 0;
}
int fdevent_event_del(fdevents *ev,int fd){
struct epoll_event ep;
fdnode *fdn=ev->fdarray[fd];
if(-1==fdn->status){
return 0;
}
memset(&ep,0,sizeof(ep));
ep.data.ptr=NULL;
ep.data.fd=fd;
if(0 != epoll_ctl(ev->epoll_fd,EPOLL_CTL_DEL,fd,&ep)){
fprintf(stderr,"epoll del failed %s\n",strerror(errno));
return -1;
}
fdn->status=-1;
return 0;
}
int fdevent_poll(fdevents *ev, int timeout_ms){
int n,i;
for(;;){
n=0;
if((n=epoll_wait(ev->epoll_fd,ev->epoll_events,ev->maxfds,timeout_ms))>0){
for(i=0;i<n;i++){
fdevent_handler handler;
void *context;
int events = 0, e,fd;
fdnode *fdn;
e=ev->epoll_events[i].events;
if (e & EPOLLIN) events |= FDEVENT_IN;
if (e & EPOLLOUT) events |= FDEVENT_OUT;
fd=ev->epoll_events[i].data.fd;
fdn = ev->fdarray[fd];
context = fdn->ctx;
fdn->handler(fd,context,events);
}
}
}
}
static fdnode *fdnode_init(){
fdnode *fdn;
fdn=calloc(1,sizeof(*fdn));
fdn->fd=-1;
fdn->status=-1;
return fdn;
}
static void fdnode_free(fdnode *fdn){
free(fdn);
}
简单调用流程就是下面这几个接口
fdevents *fdevent_init(size_t maxfds);
int fdevent_register(fdevents *ev,int fd,fdevent_handler handler,void *ctx);
int fdevent_event_add(fdevents *ev,int fd,int events);
int fdevent_poll(fdevents *ev, int timeout_ms);
分享到:
相关推荐
在本部分我们实现一个较为好用实用的Epoll并发类, 由于实现代码与使用方式较简单, 因此就不在此赘述了, 下面我还使用了该类实现了一个基于Epoll的echo-server, 以演示该类的用法; 由于此处仅为Epoll类库的第一个...
封装了epoll功能,很简单的使用epoll,只需要指定epoll服务监听端口,就可以运行epoll服务。可以指定收到数据的回调函数,可以指定有连接的回调函数,可以指定网络关闭的回调函数。自己编写调试epoll模型可能要几天...
`pi-pi-net`是一个专为Linux环境设计的网络库,它封装了Linux系统中的`epoll`机制,为开发者提供了构建高效、可扩展的网络应用程序的便利工具。这个库的核心价值在于简化了Reactor模式的实现,使得开发Web服务器、...
iocp-epoll的几个例子,提供参考。 其中,iocp包含了几种使用方式,同步异步处理方式, epoll为linux下面使用,编译时可以使用下面简单命令, g++ -p -g -o hhh ***.cpp -l pthread
epoll的接口非常简单,一共就三个函数: 1. int epoll_create(int size); 2. int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event) 3. int epoll_wait(int epfd, struct epoll_event * events, int...
C++中封装`epoll`,可以创建一个`EpollServer`类,包含上述步骤的成员函数,如`createSocket`、`bindAndListen`、`initEpoll`、`registerSocket`、`acceptConnection`、`handleEvents`等。此外,可以考虑使用智能...
前言 但golang有gc,而且基本可以认为是抢占式调度,golang的性能与c ++部分还是差很多!我心目中的c ++协程库应该是...使用boost.Coroutine2对epoll系统调用的简单封装 使用说明 1.下载升压 到官网下载最新的发行版
以下是一个简单的epoll使用示例: ```cpp #include <sys/epoll.h> int epoll_fd = epoll_create1(0); struct epoll_event event; event.events = EPOLLIN; event.data.fd = conn->socket(); if (epoll_ctl(epoll_...
本文将深入探讨如何将`socket select`模型简单地封装为类,并结合`粘包`和`组包`的概念,实现进程间通信。 首先,`socket`是连接网络应用的桥梁,它允许程序在本地主机上创建一个端点,并通过这个端点与其他主机上...
本篇文章将深入探讨如何对`curl_multi`进行简单封装,以创建一个高效、实用的PHP库。 1. **理解curl_multi**: `curl_multi`接口允许我们同时处理多个cURL句柄,而不是逐个执行。这通过利用底层系统级别的多路复用...
总结来说,"对ACE的Proactor通讯模式的全面封装"是一项旨在提供简单易用的异步通信接口的工作,它隐藏了复杂的ACE库细节,让开发者能更专注于应用层的开发。通过封装后的ACESVC组件,我们可以更高效地构建高性能网络...
首先,让我们了解一下epoll的基本概念。epoll是对传统的select和poll模型的改进,它采用“事件驱动”模型,通过内核与用户空间的高效交互,实现了对大量文件描述符的管理和监控。epoll的核心操作包括`epoll_create`...
总结来说,libevent是一个强大的事件驱动库,它将复杂的底层IO复用机制封装起来,为开发者提供了统一的接口,使得编写高性能、高并发的服务程序变得更加简单。如果你已经具备了服务并发编程的基础,了解了select和...
了解网络包的封装和解封装,以及IP地址、端口号、协议头等概念,将使你能够解决更复杂的问题。 总的来说,Linux网络编程是一个涵盖广泛且深入的领域,不仅需要掌握底层协议的工作原理,还要熟悉上层应用程序的设计...
使用epoll+线程池+异步网络IO模式开发,并发性能优越。 代码经过大量的稳定性、性能测试,可满足商用服务器项目。 支持linux、macos、ios、android、windows平台 特性: 网络库 tcp/udp客户端,接口简单易用并且是...
本文将深入探讨这两个主题,并结合C++编程语言,阐述如何在实际项目中实现一个简单的线程池框架。 首先,线程池是一种线程管理策略,它预先创建一组线程,当需要执行任务时,任务被放入队列,由线程池中的空闲线程...
多路分离复用:是对Epoll和IO的使用封装,使得使用者可以完全与Epoll的细节隔开,专心于IO的处理 线程池:提供一种执行任务的服务,用户只需按照任务接口实现自己任务然后将它放到线程池即可执行任务 内存管理:实现...
对于高并发场景,通常需要使用多线程或多进程处理客户端连接,或者利用异步I/O模型(如epoll在Linux下)提高效率。 7. **实例应用**: - **简单HTTP服务器**:使用Berkley套接字实现一个基本的HTTP服务器,响应...
Server类中包括了有一个重要的类叫做Eventlooper这个类就是对epoll的封装,要用epoll_ctl注册到epoll上的fd又被封装为Channel类,当有数据到来需要操作时,channel中的几个函数指针就指向了需要回调的函数, 这里...
总结来说,`libuv` 对 `epoll` 的封装让开发者能够更简单、高效地处理并发 I/O 任务,通过提供统一的跨平台接口,使得开发者可以在不关心底层操作系统差异的情况下编写高度可移植的代码。`libuv` 的事件驱动模型、...