unix网络编程(四)epoll反应堆

news2024/11/15 23:23:26

epoll反应堆

  • 原理
  • 代码
    • demo

原理

【epoll模型原来的流程】:

epoll_create(); // 创建监听红黑树
epoll_ctl(); // 向书上添加监听fd
epoll_wait(); // 监听
有监听fd事件发送—>返回监听满足数组—>判断返回数组元素—>
lfd满足accept—>返回cfd---->read()读数据—>write()给客户端回应。

【epoll反应堆模型的流程】:

epoll_create(); // 创建监听红黑树
epoll_ctl(); // 向书上添加监听fd
epoll_wait(); // 监听
有客户端连接上来—>lfd调用acceptconn()—>将cfd挂载到红黑树上监听其读事件—>
epoll_wait()返回cfd—>cfd回调recvdata()—>将cfd摘下来监听写事件—>
epoll_wait()返回cfd—>cfd回调senddata()—>将cfd摘下来监听读事件—>…—>

在这里插入图片描述
参考青萍之末的博客,其中有具体的epoll和epoll反应堆模型的分析。
参考B站学习视频,Linux系统编程和linux网络编程教程。

代码

demo

实现一个epoll反应堆模型的反射服务器。

#include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<sys/epoll.h>
#include<libgen.h>
#include "pub.h"

#define MAXSIZE 1024
#define MAXOPEN 1024

// 自定义事件驱动结构体,必包括fd、void *、call_back,按情况添加其他的数据
typedef struct _xevent
{
	int fd;
	uint32_t events;
	int epfd;
	void *arg;
	void (*call_back)(void *arg1, void *arg2);  // 第一个为xevent的项,第二个为xevent数组的地址

	char buf[MAXSIZE];
	int buflen;
} xevent;

void write_data(void *arg1, void *arg2);
void read_data(void *arg1, void *arg2);

// 注册事件
void eventadd(int epfd, int fd, uint32_t events, xevent *xev, void (*call_back)(void *, void *))
{
	// 初始化xevent
	xev->fd = fd;
	xev->events = events;
	xev->epfd = epfd;
	xev->call_back = call_back;
	
	// 初始化epoll_event , 上树监听
	struct epoll_event ev;
	ev.events = events;
	ev.data.ptr = xev;

	epoll_ctl(epfd, EPOLL_CTL_ADD, fd, &ev);
}

void eventmod(int epfd, int fd, uint32_t events, xevent *xev, void(*call_back)(void *, void *))
{
	// 修改xevent结构体
	xev->fd = fd;
	xev->events = events;
	xev->epfd = epfd;
	xev->call_back = call_back;

	// 修改epoll_event
	struct epoll_event ev;
	ev.events= events;
	ev.data.ptr = xev;
	epoll_ctl(epfd, EPOLL_CTL_MOD, fd, &ev);
}

// 删除事件
void eventdel(int epfd, int fd, xevent *xev)
{
	// 修改xevent结构体
	xev->fd = 0;
	xev->events = 0;
	xev->epfd = 0;
	xev->call_back = NULL;

	// 删除epoll_event
	epoll_ctl(epfd, EPOLL_CTL_DEL, fd, NULL);
}

// 修改事件
/* 对本程序而言,功能为接受数据并将接受的数据发送回去,因此可以
 * 在接收数据时将fd改为监听可写,在发送数据后改为监听可读。
 * 通常在需要监听可写时,注册fd的可写事件,写完即注销fd的可写事件,
 * 需要来回上树下树十分浪费时间,本程序是取巧的行为
 */
// 读数据
void read_data(void *arg1, void *arg2)
{
	xevent *xev = arg1;
	xevent *my_xevents = arg2;

	xev->buflen = read(xev->fd, xev->buf, sizeof(xev->buf));
	if(xev->buflen > 0)
	{
		// 读到数据,设置可写监听
		eventmod(xev->epfd, xev->fd, EPOLLOUT, xev, write_data);
	}
	else if(0 == xev->buflen)
	{
		// 客户端关闭,关闭fd、删除fd监听事件
		printf("client close\n");
		close(xev->fd);
		eventdel(xev->epfd, xev->fd, xev);
	}
	else
	{
		perror("read");
		exit(-1);
	}
}

//写数据
void write_data(void *arg1, void *arg2)
{
	xevent *xev = arg1;
	xevent *my_xevents = arg2;
	
	// 写数据,并设置可读监听
	write(xev->fd, xev->buf, xev->buflen);
	xev->buflen = 0;
	
	eventmod(xev->epfd, xev->fd, EPOLLIN, xev, read_data);
}

// 接受新连接
void init_accept(void *arg1, void *arg2)
{
	xevent *xev = arg1;
	xevent *my_xevents = arg2;

	// 从my_xevents数组中找出一个未使用的xevent,accept新连接并上树监听
	for(int i = 0; i < MAXOPEN; i++)
	{
		if(my_xevents[i].fd == 0)
		{
			int cfd = Accept_with_print(xev->fd);
			eventadd(xev->epfd, cfd, EPOLLIN, &my_xevents[i], read_data);
			break;
		}
	}
}
int main(int argc, char *argv[])
{
	if(argc < 3)
	{
		printf("usage: ./%s ip_address port_number\n", basename(argv[0]));
		return -1;
	}
	char *ip = argv[1];
	int port = atoi(argv[2]);

	int lfd = socketBind(ip, port);

	// 创建内核事件表的描述符
	int epfd = epoll_create(MAXOPEN);
	if(-1 == epfd)
	{
		perror("epoll_create");
		return -1;
	}

	// 创建xevent数组并初始化
	xevent my_xevents[MAXOPEN];
	memset(my_xevents, 0, sizeof(my_xevents));

	// 创建接受内核事件通知的epoll_event数组
	struct epoll_event events[MAXOPEN];

	// lfd注册监听,使用xevent数组的最后一个
	eventadd(epfd, lfd, EPOLLIN, &my_xevents[MAXOPEN-1], init_accept);

	while(1)
	{
		int nready = epoll_wait(epfd, events, MAXOPEN, -1);
		if(nready < 0)
		{
			perror("epoll_wait");
			break;
		}
		for(int i = 0; i<nready; i++)
		{
			xevent *xev = events[i].data.ptr;
			if(events[i].events == xev->events)
			{
				xev->call_back(xev, my_xevents);
			}
		}

	}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/23155.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【仿牛客网笔记】项目进阶,构建安全高效的企业服务——任务执行和调度

http://www.quartz-scheduler.org 集群服务器有两类程序一类普通请求&#xff0c;另一类定时任务。 浏览器通过负载均衡工具Nginx, Nginx同一时刻有一个去处理。 Quartz存入到数据库中&#xff0c;多个Quartz同时访问 线程池测试类 实例化logger,通过logger输出&#xff0c;自…

Java学习之封装

**学习重点&#xff1a;**面向对象编程的三大特征&#xff1a;封装、继承、多态度 封装介绍 封装&#xff1a;encapsulation就是把属性和方法封装在一起&#xff0c;数据被保护在内部&#xff0c;程序的其它部分只有通过被授权的操作&#xff08;方法&#xff09;&#xff0c…

【Linux】四、Linux 进程概念(二)|普遍操作系统的进程状态(运行、阻塞、挂起)|Linux的具体进程状态|僵尸进程|孤儿进程

目录 五、进程状态 5.1 普遍操作系统层面的进程状态&#xff08;宏观&#xff09; 5.1.1 什么叫做运行状态 5.1.2 阻塞状态 5.1.2 挂起状态 5.2 Linux 的进程状态&#xff08;具体&#xff09; 5.2.1 Linux 内核进程状态源代码 5.2.2 Linux 进程状态查看 5.2.3 Linux …

2023年软考高级系统规划与与管理师备考策略

2023年上半年软考高级系统规划与管理师打算怎么备考呢&#xff1f; 系统规划与管理师是高级科目的其中一科&#xff0c;难度系数是有的&#xff0c;系统规划与管理师相较于其他高级考试又相对难度可控。只要你肯下功夫&#xff0c;还是可以拿下的&#xff01;系统规划与管理师…

ASP.NET Core教程-Routing(路由)

更新记录 转载请注明出处&#xff1a; 2022年11月21日 发布。 2022年11月20日 从笔记迁移到博客。 Route Basic(路由基础) 路由说明 请求URL映射到控制器方法的过程&#xff0c;这个映射过程由路由规则定义。 路由功能 跟据预先配置的路由信息对客户端传来的请求进行路由映射&a…

Hbase Coprocessor 协处理器 与 JavaAPI

协处理器概念 一、协处理器有两种&#xff1a; observer 和 endpoint 1、observer协处理器 Observer 类似于传统数据库中的触发器&#xff0c;当发生某些事件的时候这类协处理器会被 Server 端调用。Observer Coprocessor 就是一些散布在 HBase Server 端代码中的 hook 钩子…

go使用grpc

protoc下载 protoc是protobuf的编译工具&#xff0c;能根据.proto文件生成为各种语言的文件。 原始的protoc集成了如下语言的转换&#xff1a; cc#javaobjectcphppythonruby 但是没有集成go的转换工具。go的转换工具是在protoc的基础上使用插件的方式运行。 protoc 的下载地址…

java面试强基(7)

hashCode() 有什么用&#xff1f; ​ hashCode() 的作用是获取哈希码&#xff08;int 整数&#xff09;&#xff0c;也称为散列码。这个哈希码的作用是确定该对象在哈希表中的索引位置。 ​ hashCode()定义在 JDK 的 Object 类中&#xff0c;这就意味着 Java 中的任何类都包含…

阿里架构师耗时1年,把P8所需要的整个Java体系,都整理到了一起

始末 有人调侃我们说&#xff1a; 程序员不如送外卖。送外卖是搬运食物&#xff0c;自己是搬运代码&#xff0c;都不产出新的东西……透支体力&#xff0c;又消耗健康&#xff0c;可替代性极强&#xff0c;30岁之后就要面临被优化的危险……想跳槽&#xff0c;但是更高的平台…

科技风杂志科技风杂志社科技风编辑部2022年第31期目录

科技创新《科技风》投稿&#xff1a;cnqikantg126.com 基于钢架网箱清理控制系统的研究 张浩;崔新忠;丁树兵;尹杰;卢佳乐;王隽屹;张津宁; 1-3 遥感技术在海洋区域地质调查中的应用——以岛礁区调查为例 韩艳飞; 4-6 医院电气工程智能自动化及PLC技术在医院电气设备…

mipi介绍

文章目录1.MIPI简介1-1&#xff1a;DSI layer2.D_PHY2-1&#xff1a;D_PHY介绍2-2&#xff1a;电平状态2-3&#xff1a;lane结构2-4&#xff1a;data lane操作模式2-4-1&#xff1a;escape mode和space one hot coding2-4-2&#xff1a;Low-Power Data Transmission&#xff08…

图书借阅管理系统的设计与实现/书籍借还管理系统

摘要 随着Internet的发展&#xff0c;人们的日常生活已经离不开网络。未来人们的生活与工作将变得越来越数字化、网络化和电子化。网上借阅一系列功能将成为人们最关注话题&#xff0c;本论文是以构建图书借阅为目标&#xff0c;使用 Java制作&#xff0c;由前台用户借阅图书、…

计算机网络4小时速成:网络安全,被动攻击,主动攻击,对称加密,公钥秘钥,数字签名,鉴别,网络层安全协议IPsec,传输层安全协议SSL,防火墙,入侵检测系统

计算机网络4小时速成&#xff1a;网络安全&#xff0c;被动攻击&#xff0c;主动攻击&#xff0c;对称加密&#xff0c;公钥秘钥&#xff0c;数字签名&#xff0c;鉴别&#xff0c;网络层安全协议IPsec&#xff0c;传输层安全协议SSL&#xff0c;防火墙&#xff0c;入侵检测系统…

PTA题目 计算分段函数[1]

本题目要求计算下列分段函数f(x)的值&#xff1a; 输入格式: 输入在一行中给出实数x。 输出格式: 在一行中按“f(x) result”的格式输出&#xff0c;其中x与result都保留一位小数。 输入样例1: 10输出样例1: f(10.0) 0.1输入样例2: 0输出样例2: f(0.0) 0.0 #includ…

公众号免费题库接口

公众号免费题库接口 本平台优点&#xff1a; 多题库查题、独立后台、响应速度快、全网平台可查、功能最全&#xff01; 1.想要给自己的公众号获得查题接口&#xff0c;只需要两步&#xff01; 2.题库&#xff1a; 查题校园题库&#xff1a;查题校园题库后台&#xff08;点击…

IDEA Windows下SPARK连接Hive

IDEA Windows下SPARK连接Hive 文章目录IDEA Windows下SPARK连接Hive一、本地Windows环境配置二、IDEA项目配置1. POM配置2. 资源文件配置3. 测试验证一、本地Windows环境配置 本地构建HADOOP客户端 将大数据平台的HAODOP环境打包拿到本地环境来&#xff1a; #压缩整个HADOOP…

特征选择技术总结

在本文中&#xff0c;我们将回顾特性选择技术并回答为什么它很重要以及如何使用python实现它。 本文还可以帮助你解答以下的面试问题: 什么是特征选择?说出特性选择的一些好处你知道哪些特征选择技巧?区分单变量、双变量和多变量分析。我们能用PCA来进行特征选择吗?前向特征…

如何打造一支专业的QA团队,至少要关注这5点

目录 第一、专职的QA人员 第二、关注QA的能力素质 第三、适度的量化指标 第四、增加QA工作的透明度 第五、 充分利用好评审这把“利剑” QA是Quality Assurance的缩写&#xff0c;直接翻译过来就是“质量保证”的意思。 QA也称为质量工程师&#xff0c;基本上每个产品型或…

指纹浏览器——为跨境电商打造的跨境卫士

“跨境卫士”这个词当然是保障跨境人账号安全的意思&#xff0c;为跨境电商打造的跨境卫士&#xff0c;说白了就是专门为跨境电商商家解决各种痛点的软件工具&#xff0c;那一说到这类软件工具&#xff0c;很多人脑海中第一时间想到的是什么&#xff1f;相信都会出现“指纹浏览…

zabbix日志监控:操作系统、业务系统、文件大小、多行日志

zabbix日志监控&#xff1a;操作系统、业务系统、文件大小、多行日志 目录1 监控操作系统日志2 监控业务系统日志具体要求&#xff1a;分析&#xff1a;操作&#xff1a;3 监控日志文件大小&#xff08;1&#xff09;在被管主机当中安装agent&#xff08;2&#xff09;在以下za…