Linux-统一事件源
概念:
信号时一种异步事件:信号处理函数和程序的主循环式两条不同的执行路线,信号处理函数需要尽可能快地执行完毕,以确保该信号不被屏蔽.(为了避免一些竞态条件,信号在处理期间,系统不会再次出发它)太久.这里就采用一种常用的解决方案是:把信号的主要处理函数逻辑放到程序的主循环中,当信号处理函数被触发时,它只是简单地通知主循环程序接受到信号,并把信号值传递给主函数.主循环在根据接受到的信号值执行目标信号对应的处理逻辑代码.通常采用管道的方式来将"信号"传递给主循环.主程序采用I/O复用模型来将信号事件和其他事件统一处理.即统一事件源.
代码实例:
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <fcntl.h>
#include <sys/epoll.h>
#include <pthread.h>
#include <errno.h>
#include <signal.h> #define MAX_EVENT_NUMBER 1024
static int pipefd[]; int setnonblocking(int fd){
int old_option=fcntl(fd,F_GETFL);
int new_option=old_option | O_NONBLOCK; fcntl(fd,F_SETFL,new_option);
return old_option;
} void addfd(int epollfd,int fd){
struct epoll_event event;
event.data.fd=fd;
event.events =EPOLLIN | EPOLLET;
epoll_ctl(epollfd,EPOLL_CTL_ADD,fd,&event);
setnonblocking(fd);
} void sig_handler(int sig){
int save_errno=errno;
int msg=sig;
send(pipefd[],(char*)&msg,,);
errno=save_errno;
} void addsig(int sig){
struct sigaction sa;
memset(&sa,'\0',sizeof(sa));
sa.sa_flags|=SA_RESTART;
sa.sa_handler=sig_handler;
sigfillset(&sa.sa_mask);
assert(sigaction(sig,&sa,NULL)!=-);
} int main(int argc,char* argv[]){
if(argc<=){
printf("usage:%s ip_address port_number\n",argv[]);
return -;
} const char* ip=argv[];
int port=atoi(argv[]); struct sockaddr_in server_addr;
memset(&server_addr,,sizeof(server_addr));
server_addr.sin_family=AF_INET;
inet_pton(AF_INET,ip,&server_addr.sin_addr));
server_addr.sin_port=htons(port)); int listenfd=socket(AF_INET,SOCK_STREAM,);
assert(listenfd!=-); int ret=bind(listenfd,(struct sockaddr*)&server_addr,sizeof(server_addr));
assert(ret!=-); ret=listen(listenfd,);
assert(ret!=-); epoll_event events[MAX_EVENT_NUMBER];
int epollfd=epoll_create();
assert(epollfd!=-);
addfd(epollfd,listenfd); ret=socketpair(AF_UNIX,SOCK_STREAM,,pipefd);
assert(ret!=-); setnonblocking(pipefd[]);
addfd(epollfd,pipefd[]); addsig(SIGHUP);
addsig(SIGCHLD);
addsig(SIGTERM);
addsig(SIGINT);
bool stop_server=false; while(!stop_server){
int number=epoll_wait(epollfd,events,MAX_EVENT_NUMBER,-);
if((number<) && (errno!=EINTR)){
printf("epoll failure\n");
break;
} else{
for(int i=;i<number;i++){
int sockfd=events[i].data.fd;
if(sockfd==listenfd){
struct sockaddr_in client;
bzero(&client,sizeof(client));
socklen_t len=sizeof(client); int connfd=accept(sockfd,(struct sockaddr*)&client,&len);
addfd(epollfd,connfd);
} else if(sockfd==pipefd[] && events[i].events & EPOLLIN){
char signals[];
memset(signals,'\0',sizeof(signals)); int ret=recv(sockfd,signals,,);
if(ret<){
continue;
}
else if(ret==){
continue;
} else{
for(int i=;i<ret;i++){
switch(signals[i])){
case SIGCHLD:
case SIGHUP:
{
continue;
} case SIGTERM:
case SIGINT:
{
stop_server=true;
}
}
}
}
} else{
}
}
}
}
printf("close fds\n");
close(listenfd);
close(pipefd[]);
close(pipefd[]);
return ;
}
Linux-统一事件源的更多相关文章
- 统一事件源epoll代码示例
可以将信号注册进pipe管道的写端,通过对读端的监听,来实现统一事件源. #include <sys/types.h> #include <sys/socket.h> #inc ...
- study notes: high performance linux server programming
1:linux网络API分为:socker地址API,socker基础API,网络信息API 1,socker地址API:包含IP地址和端口(ip, port).表示TCP通信的一端. 2,socke ...
- Linux高性能server编程——信号及应用
信号 信号是由用户.系统或者进程发送给目标进程的信息.以通知目标进程某个状态的改变或系统异常. Linux信号可由例如以下条件产生: 对于前台进程.用户能够通过输入特殊的终端字符来给它发送信号. ...
- 第2章 Linux系统安装(2)_Linux系统分区及文件系统
2. 系统分区 2.1 分区类型 (1)主分区:最多只能有4个 (2)扩展分区 ①最多只能有1个,主分区加扩展分区最多有4个. ②不能写入数据,只能包含逻辑分区 (3)逻辑分区 2.2 格式化(高级格 ...
- 13.linux中断处理程序
linux中断处理程序 一.中断处理流程 在linux内核代码中进入entry-armv.S目录: linux统一的入口:__irq svc. 进入了统一的入口之后,程序跳到irq_handler标号 ...
- Linux爆新漏洞,长按回车键70秒即可获得root权限
漏洞来源这个安全问题来源于Cryptsetup存在的一个漏洞(CVE-2016-4484).Cryptsetup是在Linux统一密钥设置(Linux Unified Key Setup, LUKS) ...
- linux概念之/proc与/sys
http://blog.chinaunix.net/uid-1835494-id-3070465.html proc/x:1/sched http://bbs.chinaunix.net/threa ...
- 【Linux】常用命令
一.文件结构 /: 根目录 /bin: 系统所需要的那些命令位于此目录. /boot: Linux的内核及引导系统程序所需要的文件目录,GRUB或LILO系统引导管理器也 ...
- 菜鸟必须知道的linux的文件目录结构
Linux文件目录结 / 根目录,所有的目录.文件.设备都在/之下,/就是Linux文件系统的组织者,也是最上级的领导者. /bin bin就是二进制(binary)英文缩写.在一般的系统当中,你都可 ...
随机推荐
- jQuery异步加载数据添加事件
几个月前在一个项目中涉及到树形栏,然后看了很多插件,觉得有点麻烦,于是自己写了一个,写着写着就出问题了. 当时项目是通过树形栏进行权限控制的,管理员可以对从数据库去的数据动态生成树形栏进行增删改查操作 ...
- 关于在transform下的子元素设置fixed无效的困惑
最近的项目是要实现一个点击显示隐藏边栏的效果,而且需要边栏随着滚动而滚动. 思路简单,不就一个css的动画和一个position为fixed,搞定!但不想,设为fixed的子元素竟然无法fixed,这 ...
- smarty练习: 设置试题及打印试卷
数据库表格:shiti, shititimu, timu, kemu, xuanxiang 根据科目设置一个可以添加试题的页面:(如下图) 具体的题目从数据库中查出并形成一张试卷的形式 考试试题设置: ...
- HTML标签区别
一.div和span的区别 div是一个块级元素,可以包含段落,表格等内容,用于放置不同的内容.一般我们在网页通过div来布局定位网页中的每个区块. span是一个内联元素,没有实际意义,它的存在纯粹 ...
- canvas加载图像
之前按照例子,加载图像,发现加载图像,加载不上去,代码也没有错误,经过几次的尝试:发现需要重复调用下drawImage才可以. <script type="text/javascrip ...
- android系统自带的Service原理与使用
1. 说明 android的后台运行在很多service,它们在系统启动时被SystemServer开启,支持系统的正常工作,比如 MountService监听是否有SD卡安装及移除,Clipboar ...
- POJ 3384 Feng Shui
http://poj.org/problem?id=3384 题意:给一个凸包,求往里面放两个圆(可重叠)的最大面积时的两个圆心坐标. 思路:先把凸包边往内推R,做半平面交,然后做旋转卡壳,此时得到最 ...
- T-SQL函数类别统计
- poj3708:函数式化简+高精度进制转换+同余方程组
题目大意 给定一个函数 找出满足条件 等于 k 的最小的x m,k,d已知 其中 m,k 很大需要使用高精度存储 思路: 对 函数f(m)进行化简 ,令t=ceil( log(d,m) ) 可以得 ...
- URAL 2038 Minimum Vertex Cover
2038. Minimum Vertex Cover Time limit: 1.0 secondMemory limit: 64 MB A vertex cover of a graph is a ...