当前位置: 首页 > news >正文

linux c epoll mysql_epoll简介和使用

epoll简介

在linux的网络编程中,很长的时间都在使用select来做事件触发。在linux新的内核中,有了一种替换它的机制,就是epoll。

相比于select,epoll最大的好处在于它不会随着监听fd数目的增长而降低效率。因为在内核中的select实现中,它是采用轮询来处理的,轮询的fd数目越多,自然耗时越多。并且,在linux/posix_types.h头文件有这样的声明:

#define __FD_SETSIZE    1024

表示select最多同时监听1024个fd,当然,可以通过修改头文件再重编译内核来扩大这个数目,但这似乎并不治本。

epoll的接口非常简单,一共就三个函数:

1. int epoll_create(int size);

创建一个epoll的句柄,size用来告诉内核这个监听的数目一共有多大。这个参数不同于select()中的第一个参数,给出最大监听的fd+1的值。需要注意的是,当创建好epoll句柄后,它就是会占用一个fd值,在linux下如果查看/proc/进程id/fd/,是能够看到这个fd的,所以在使用完epoll后,必须调用close()关闭,否则可能导致fd被耗尽。

2. int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);

epoll的事件注册函数,它不同与select()是在监听事件时告诉内核要监听什么类型的事件,而是在这里先注册要监听的事件类型。第一个参数是epoll_create()的返回值,第二个参数表示动作,用三个宏来表示:

EPOLL_CTL_ADD:注册新的fd到epfd中;

EPOLL_CTL_MOD:修改已经注册的fd的监听事件;

EPOLL_CTL_DEL:从epfd中删除一个fd;

第三个参数是需要监听的fd,第四个参数是告诉内核需要监听什么事,struct epoll_event结构如下:

struct epoll_event {

__uint32_t events;  /* Epoll events */

epoll_data_t data;  /* User data variable */

};

events可以是以下几个宏的集合:

EPOLLIN :表示对应的文件描述符可以读(包括对端SOCKET正常关闭);

EPOLLOUT:表示对应的文件描述符可以写;

EPOLLPRI:表示对应的文件描述符有紧急的数据可读(这里应该表示有带外数据到来);

EPOLLERR:表示对应的文件描述符发生错误;

EPOLLHUP:表示对应的文件描述符被挂断;

EPOLLET: 将EPOLL设为边缘触发(Edge Triggered)模式,这是相对于水平触发(Level Triggered)来说的。

EPOLLONESHOT:只监听一次事件,当监听完这次事件之后,如果还需要继续监听这个socket的话,需要再次把这个socket加入到EPOLL队列里

3. int epoll_wait(int epfd, struct epoll_event * events, int maxevents, int timeout);

等待事件的产生,类似于select()调用。参数events用来从内核得到事件的集合,maxevents告之内核这个events有多大,这个maxevents的值不能大于创建epoll_create()时的size,参数timeout是超时时间(毫秒,0会立即返回,-1将不确定,也有说法说是永久阻塞)。该函数返回需要处理的事件数目,如返回0表示已超时。

下面是我在redhat9上用epoll实现的简单的C/S通信,已经运行通过了。

server.c

#include

#include

#include

#include

#include

#include

#include

#include

#define BUFFER_SIZE 40

#define MAX_EVENTS 10

int main(int argc, char * argv[])

{

int server_sockfd;// 服务器端套接字

int client_sockfd;// 客户端套接字

int len;

struct sockaddr_in my_addr;   // 服务器网络地址结构体

struct sockaddr_in remote_addr; // 客户端网络地址结构体

int sin_size;

char buf[BUFFER_SIZE];  // 数据传送的缓冲区

memset(&my_addr,0,sizeof(my_addr)); // 数据初始化--清零

my_addr.sin_family=AF_INET; // 设置为IP通信

my_addr.sin_addr.s_addr=INADDR_ANY;// 服务器IP地址--允许连接到所有本地地址上

my_addr.sin_port=htons(8000); // 服务器端口号

// 创建服务器端套接字--IPv4协议,面向连接通信,TCP协议

if((server_sockfd=socket(PF_INET,SOCK_STREAM,0))<0)

{

perror("socket");

return 1;

}

// 将套接字绑定到服务器的网络地址上

if (bind(server_sockfd,(struct sockaddr *)&my_addr,sizeof(struct sockaddr))<0)

{

perror("bind");

return 1;

}

// 监听连接请求--监听队列长度为5

listen(server_sockfd,5);

sin_size=sizeof(struct sockaddr_in);

// 创建一个epoll句柄

int epoll_fd;

epoll_fd=epoll_create(MAX_EVENTS);

if(epoll_fd==-1)

{

perror("epoll_create failed");

exit(EXIT_FAILURE);

}

struct epoll_event ev;// epoll事件结构体

struct epoll_event events[MAX_EVENTS];// 事件监听队列

ev.events=EPOLLIN;

ev.data.fd=server_sockfd;

// 向epoll注册server_sockfd监听事件

if(epoll_ctl(epoll_fd,EPOLL_CTL_ADD,server_sockfd,&ev)==-1)

{

perror("epll_ctl:server_sockfd register failed");

exit(EXIT_FAILURE);

}

int nfds;// epoll监听事件发生的个数

// 循环接受客户端请求

while(1)

{

// 等待事件发生

nfds=epoll_wait(epoll_fd,events,MAX_EVENTS,-1);

if(nfds==-1)

{

perror("start epoll_wait failed");

exit(EXIT_FAILURE);

}

int i;

for(i=0;i{

// 客户端有新的连接请求

if(events[i].data.fd==server_sockfd)

{

// 等待客户端连接请求到达

if((client_sockfd=accept(server_sockfd,(struct sockaddr *)&remote_addr,&sin_size))<0)

{

perror("accept client_sockfd failed");

exit(EXIT_FAILURE);

}

// 向epoll注册client_sockfd监听事件

ev.events=EPOLLIN;

ev.data.fd=client_sockfd;

if(epoll_ctl(epoll_fd,EPOLL_CTL_ADD,client_sockfd,&ev)==-1)

{

perror("epoll_ctl:client_sockfd register failed");

exit(EXIT_FAILURE);

}

printf("accept client %s/n",inet_ntoa(remote_addr.sin_addr));

}

// 客户端有数据发送过来

else

{

len=recv(client_sockfd,buf,BUFFER_SIZE,0);

if(len<0)

{

perror("receive from client failed");

exit(EXIT_FAILURE);

}

printf("receive from client:%s",buf);

send(client_sockfd,"I have received your message.",30,0);

}

}

}

return 0;

}

client.c

#include

#include

#include

#include

#include

#include

#include

#define BUFFER_SIZE 40

int main(int argc, char *argv[])

{

int client_sockfd;

int len;

struct sockaddr_in remote_addr; // 服务器端网络地址结构体

char buf[BUFFER_SIZE];  // 数据传送的缓冲区

memset(&remote_addr,0,sizeof(remote_addr)); // 数据初始化--清零

remote_addr.sin_family=AF_INET; // 设置为IP通信

remote_addr.sin_addr.s_addr=inet_addr("127.0.0.1");// 服务器IP地址

remote_addr.sin_port=htons(8000); // 服务器端口号

// 创建客户端套接字--IPv4协议,面向连接通信,TCP协议

if((client_sockfd=socket(PF_INET,SOCK_STREAM,0))<0)

{

perror("client socket creation failed");

exit(EXIT_FAILURE);

}

// 将套接字绑定到服务器的网络地址上

if(connect(client_sockfd,(struct sockaddr *)&remote_addr,sizeof(struct sockaddr))<0)

{

perror("connect to server failed");

exit(EXIT_FAILURE);

}

// 循环监听服务器请求

while(1)

{

printf("Please input the message:");

scanf("%s",buf);

// exit

if(strcmp(buf,"exit")==0)

{

break;

}

send(client_sockfd,buf,BUFFER_SIZE,0);

// 接收服务器端信息

len=recv(client_sockfd,buf,BUFFER_SIZE,0);

printf("receive from server:%s/n",buf);

if(len<0)

{

perror("receive from server failed");

exit(EXIT_FAILURE);

}

}

close(client_sockfd);// 关闭套接字

return 0;

}

makefile

#This is the makefile of EpollTest

.PHONY:all

all:server client

server:

gcc server.c -o server

client:

gcc client.c -o client

clean:

相关文章:

  • python可以不对变量初始化_Python的诡异陷阱
  • python怎么绘制图例_用Python绘制图例
  • centos7安装mysql5.7.23_centos7.1上安装Mysql5.7.23
  • java客户端_大家都是怎么发布Java客户端程序的?难道让用户自己装JRE?
  • java 加载jar_java – 在运行时加载jar
  • java ant linux_linux下ant的安装
  • java获取数据库信息_java获取数据库的库、表、字段信息
  • java jpanel 布局管理器,JPanel和Java布局管理器
  • java 验证码识别 类库_iCaptcha
  • java调用wsdl证书错误_java调用.net web service服务,报证书错误
  • mysql 联表删除limit_MySQL联结多表更新和删除_mysql
  • mysql数据库备份总结_mysql中mysqlhotcopy备份数据库总结
  • java 反射 api_七分钟理解 Java 的反射 API
  • java 导入dbf文件_用java怎么实现批量导入dbf文件(从dbf文件导入到sqlserver2008中)到sqlserver2008...
  • md5算法 java_MD5算法[ java 实现]
  • C++回声服务器_9-epoll边缘触发模式版本服务器
  • CSS进阶篇--用CSS开启硬件加速来提高网站性能
  • Debian下无root权限使用Python访问Oracle
  • HomeBrew常规使用教程
  • iOS | NSProxy
  • Java-详解HashMap
  • JS基础之数据类型、对象、原型、原型链、继承
  • miaov-React 最佳入门
  • Mysql5.6主从复制
  • RxJS 实现摩斯密码(Morse) 【内附脑图】
  • uni-app项目数字滚动
  • Web设计流程优化:网页效果图设计新思路
  • 从tcpdump抓包看TCP/IP协议
  • 仿天猫超市收藏抛物线动画工具库
  • 巧用 TypeScript (一)
  • 实习面试笔记
  • 算法之不定期更新(一)(2018-04-12)
  • 腾讯大梁:DevOps最后一棒,有效构建海量运营的持续反馈能力
  • 为视图添加丝滑的水波纹
  • 我从编程教室毕业
  • 与 ConTeXt MkIV 官方文档的接驳
  • 新年再起“裁员潮”,“钢铁侠”马斯克要一举裁掉SpaceX 600余名员工 ...
  • ​Distil-Whisper:比Whisper快6倍,体积小50%的语音识别模型
  • ​LeetCode解法汇总2808. 使循环数组所有元素相等的最少秒数
  • ​业务双活的数据切换思路设计(下)
  • #我与Java虚拟机的故事#连载02:“小蓝”陪伴的日日夜夜
  • $Django python中使用redis, django中使用(封装了),redis开启事务(管道)
  • (10)ATF MMU转换表
  • (转)fock函数详解
  • ***通过什么方式***网吧
  • *上位机的定义
  • .Net Core与存储过程(一)
  • .Net 中Partitioner static与dynamic的性能对比
  • .NET/C# 使窗口永不获得焦点
  • .NET/C# 使用反射调用含 ref 或 out 参数的方法
  • .NetCore Flurl.Http 升级到4.0后 https 无法建立SSL连接
  • .net遍历html中全部的中文,ASP.NET中遍历页面的所有button控件
  • .NET设计模式(2):单件模式(Singleton Pattern)
  • [ IO.File ] FileSystemWatcher
  • [ vulhub漏洞复现篇 ] Apache APISIX 默认密钥漏洞 CVE-2020-13945