Epoll 实例
服务端调试: [test@cs2 epoll]$ g++ epoll_server.cpp -o epoll_server -lpthread [test@cs2 epoll]$ ./epoll_server
connec_ from >> 0.0.0.0
reading!
read from client: reading!
Client close connect! 客户端调试: [test@cs2 epoll]$ g++ epoll_client.cpp -o epoll_client [test@cs2 epoll]$ ./epoll_client 127.0.0.1
input message:
Message from server: input message:@ 服务端源码:epoll_server.cpp #include <iostream>
#include <sys/socket.h>
#include <sys/epoll.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>
#include <errno.h>
#include <stdlib.h>
#include <string.h>
#include <pthread.h> #define MAXLINE 1024
#define OPEN_MAX 100
#define LISTENQ 20
#define SERV_PORT 5555
#define INFTIM 1000 //线程池任务队列结构体
struct task{
int fd; //需要读写的文件描述符
struct task *next; //下一个任务
}; //用于保存向客户端发送一次消息所需的相关数据
struct user_data{
int fd;
unsigned int n_size;
char line[MAXLINE];
}; //线程的任务函数
void * readtask(void *args);
void * writetask(void *args); //声明epoll_event结构体的变量,ev用于注册事件,数组用于回传要处理的事件
struct epoll_event ev,events[];
int epfd;
pthread_mutex_t mutex;
pthread_cond_t cond1;
struct task *readhead=NULL,*readtail=NULL,*writehead=NULL; void setnonblocking(int sock)
{
int opts;
opts=fcntl(sock, F_GETFL);
if(opts<)
{
perror("fcntl(sock,GETFL)");
exit();
}
opts = opts | O_NONBLOCK;
if(fcntl(sock, F_SETFL, opts)<)
{
perror("fcntl(sock,SETFL,opts)");
exit();
}
} int main()
{
int i, maxi, listenfd, connfd, sockfd,nfds;
pthread_t tid1,tid2;
struct task *new_task = NULL;
struct user_data *rdata = NULL;
socklen_t clilen;
pthread_mutex_init(&mutex, NULL);
pthread_cond_init(&cond1, NULL);
//初始化用于读线程池的线程,开启两个线程来完成任务,两个线程会互斥地访问任务链表
pthread_create(&tid1, NULL, readtask, NULL);
pthread_create(&tid2, NULL, readtask, NULL); //生成用于处理accept的epoll专用的文件描述符
epfd = epoll_create(); struct sockaddr_in clientaddr;
struct sockaddr_in serveraddr; listenfd = socket(AF_INET, SOCK_STREAM, );
//把socket设置为非阻塞方式
setnonblocking(listenfd);
//设置与要处理的事件相关的文件描述符
ev.data.fd = listenfd; //设置要处理的事件类型,当描述符可读时出发,出发方式为ET模式
ev.events = EPOLLIN | EPOLLET; //注册epoll事件
epoll_ctl(epfd, EPOLL_CTL_ADD, listenfd, &ev);
bzero(&serveraddr, sizeof(serveraddr));
serveraddr.sin_family = AF_INET;
const char *local_addr = "127.0.0.1";
inet_aton(local_addr, &(serveraddr.sin_addr)); //htons(SERV_PORT);
serveraddr.sin_port=htons(SERV_PORT);
bind(listenfd,(sockaddr *)&serveraddr, sizeof(serveraddr)); //开始监听
listen(listenfd, LISTENQ);
maxi = ;
while() {
//等待epoll事件的发生
nfds=epoll_wait(epfd, events, , );
//处理所发生的所有事件
for(i=; i < nfds; ++i)
{
if(events[i].data.fd==listenfd)
{
connfd = accept(listenfd,(sockaddr *)&clientaddr, &clilen);
if(connfd<)
{
perror("connfd<0");
exit();
}
setnonblocking(connfd);
const char *str = inet_ntoa(clientaddr.sin_addr);
std::cout<<"connec_ from >> " << str << std::endl;
//设置用于读操作的文件描述符
ev.data.fd=connfd;
//设置用于注测的读操作事件
ev.events=EPOLLIN | EPOLLET;
//注册ev
epoll_ctl(epfd, EPOLL_CTL_ADD, connfd, &ev);
}
else if(events[i].events & EPOLLIN)
{
printf("reading!\n");
if ( (sockfd = events[i].data.fd) < ) continue;
new_task = new task();
new_task->fd =sockfd;
new_task->next = NULL;
//添加新的读任务
pthread_mutex_lock(&mutex);
if(readhead == NULL)
{
readhead = new_task;
readtail = new_task;
}
else
{
readtail->next = new_task;
readtail = new_task;
}
//唤醒所有等待cond1条件的线程
pthread_cond_broadcast(&cond1);
pthread_mutex_unlock(&mutex);
}
else if(events[i].events & EPOLLOUT)
{
rdata=(struct user_data *)events[i].data.ptr;
sockfd = rdata->fd;
write(sockfd, rdata->line, rdata->n_size);
delete rdata;
//设置用于读操作的文件描述符
ev.data.fd=sockfd;
//设置用于注测的读操作事件
ev.events=EPOLLIN | EPOLLET;
//修改sockfd上要处理的事件为EPOLIN
epoll_ctl(epfd, EPOLL_CTL_MOD, sockfd, &ev);
}
}
}
} void * readtask(void *args)
{
int fd=-;
unsigned int n;
//用于把读出来的数据传递出去
struct user_data *data = NULL;
while(){
//互斥访问任务队列
pthread_mutex_lock(&mutex);
//等待到任务队列不为空
while(readhead == NULL)
pthread_cond_wait(&cond1, &mutex); //线程阻塞,释放互斥锁,当等待的条件等到满足时,它会再次获得互斥锁
fd = readhead->fd;
//从任务队列取出一个读任务
struct task *tmp = readhead;
readhead = readhead->next;
delete tmp;
pthread_mutex_unlock(&mutex);
data = new user_data();
data->fd=fd;
if ( (n = read(fd, data->line, MAXLINE)) < )
{
if (errno == ECONNRESET)
close(fd);
else
std::cout<<"readline error"<< std::endl; if(data != NULL) delete data;
}
else if (n == )
{
//客户端关闭了,其对应的连接套接字可能也被标记为EPOLLIN,然后服务器去读这个套接字
//结果发现读出来的内容为0,就知道客户端关闭了。
close(fd);
printf("Client close connect!\n");
if(data != NULL) delete data;
}
else
{
std::cout << "read from client: " << data->line << std::endl;
data->n_size = n;
//设置需要传递出去的数据
ev.data.ptr = data;
//设置用于注测的写操作事件
ev.events = EPOLLOUT | EPOLLET;
//修改sockfd上要处理的事件为EPOLLOUT
epoll_ctl(epfd, EPOLL_CTL_MOD, fd, &ev);
}
}
} 客户端源码:epoll_client.cpp #include <stdio.h>
#include <stdlib.h>
#include <sys/un.h>
#include <netdb.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h> int main(int argc,char *argv[])
{
int connect_fd;
int ret;
char snd_buf[];
int i;
int port;
int len;
static struct sockaddr_in srv_addr;
if(argc!=){
printf("Usage: %s server_ip_address port\n",argv[]);
return ;
}
port=atoi(argv[]);
connect_fd=socket(PF_INET,SOCK_STREAM,);
if(connect_fd<){
perror("cannot create communication socket");
return ;
}
memset(&srv_addr,,sizeof(srv_addr));
srv_addr.sin_family=AF_INET;
srv_addr.sin_addr.s_addr=inet_addr(argv[]);
srv_addr.sin_port=htons(port); ret=connect(connect_fd,(struct sockaddr*)&srv_addr,sizeof(srv_addr));
if(ret==-){
perror("cannot connect to the server");
close(connect_fd);
return ;
}
memset(snd_buf,,);
while(){
write(STDOUT_FILENO,"input message:",);
bzero(snd_buf, );
len=read(STDIN_FILENO,snd_buf,);
if(snd_buf[]=='@')
break;
if(len>)
write(connect_fd,snd_buf,len);
len=read(connect_fd,snd_buf,len);
if(len>)
printf("Message from server: %s\n",snd_buf);
}
close(connect_fd);
return ;
}//end
Epoll 实例的更多相关文章
- select,poll,epoll比较
		
除常用文件i/o外,其他常用io模型:io多路复用(select和poll系统调用)信号驱动I/Olinux专有的epoll编程接口异步io(aio),linux在glibc中提供有基于线程的 pos ...
 - Epoll,Poll,Select模型比较
		
http://blog.csdn.net/liangyuannao/article/details/7776057 先说Select: 1.Socket数量限制:该模式可操作的Socket数由FD_S ...
 - [转载] 理解 epoll 的事件触发机制
		
原文: http://weibo.com/p/1001603862394207076573?sudaref=weibo.com epoll的I/O事件触发方式有两种模式:ET(Edge Trigger ...
 - 多路复用I/O epoll()
		
epoll 是Linux内核中的一种可扩展IO事件处理机制,最早在 Linux 2.5.44内核中引入,可被用于代替POSIX select 和 poll 系统调用,并且在具有大量应用程序请求时能够获 ...
 - Linux下select, poll和epoll IO模型的详解
		
http://blog.csdn.net/tianmohust/article/details/6677985 一).Epoll 介绍 Epoll 可是当前在 Linux 下开发大规模并发网络程序的热 ...
 - 用C写一个web服务器(二) I/O多路复用之epoll
		
.container { margin-right: auto; margin-left: auto; padding-left: 15px; padding-right: 15px } .conta ...
 - Select、Poll、Epoll、 异步IO 介绍
		
一.概念相关介绍 同步IO和异步IO,阻塞IO和非阻塞IO分别是什么,到底有什么区别?不同的人在不同的上下文下给出的答案是不同的.所以先限定一下本文的上下文. 本文讨论的背景是Linux环境下的net ...
 - 基于epoll实现简单的web服务器
		
1. 简介 epoll 是 Linux 平台下特有的一种 I/O 复用模型实现,于 2002 年在 Linux kernel 2.5.44 中被引入.在 epoll 之前,Unix/Linux 平台下 ...
 - UNIX网络编程——epoll 系列函数简介、与select、poll 的区别
		
前面博客<<UNIX环境高级编程--epoll函数使用详解>>有关于epoll函数的讲解. 一.epoll 系列函数简介 #include <sys/epoll.h> ...
 
随机推荐
- 自己写的一个简单PHP采集器
			
自己写的一个简单PHP采集器 <?php //**************************************************************** $url = &q ...
 - 【转】Linux 下取进程占用 cpu/内存 最高的前10个进程
			
# Linux 下 取进程占用 cpu 最高的前10个进程ps aux|head -1;ps aux|grep -v PID|sort -rn -k +3|head # linux 下 取进程占用内存 ...
 - Java伪代码示例
			
学习并转载自https://www.cnblogs.com/z245894546/p/7535261.html import.java.大道至简.*; import.java.愚公移山.*; publ ...
 - Python中用format函数格式化字符串的用法(2.7版本讲解哦!)
			
语法 它通过{}和:来代替%.“映射”示例 通过位置 In [1]: '{0},{1}'.format('kzc',18) Out[1]: 'kzc,18' In [2]: '{},{}'.forma ...
 - Centos 6\7 防火墙入门配置
			
Centos 6 -- iptables iptables 用法: iptables (选项) (参数) 选项: -t<表>:指定要操纵的表: -A:向规则链中添加条目: -D:从规则链中 ...
 - COS-2OS结构和硬件支持
			
操作系统(Operating System,简称OS),是电子计算机系统中负责支撑应用程序运行环境以及用户操作环境的系统软件,同时也是计算机系统的核心与基石.它的职责常包括对硬件的直接监管.对各种计算 ...
 - springboot--配置文件加载顺序
			
-file:./config(内部配置) -file:./ (内部配置) -classpath:/config (外部配置) -classpath:/ (外部配置) 运维: spring -jar s ...
 - Linux查看服务和强制结束服务
			
查看服务id命令,常用的两种: a: ps -ef | grep 服务名 b: netstat -antp 找到结果名称,然后 pki ...
 - 并发-HashMap和HashTable源码分析
			
HashMap和HashTable源码分析 参考: https://blog.csdn.net/luanlouis/article/details/41576373 http://www.cnblog ...
 - LeetCode——Reverse String
			
LeetCode--Reverse String Question Write a function that takes a string as input and returns the stri ...