概念:

信号时一种异步事件:信号处理函数和程序的主循环式两条不同的执行路线,信号处理函数需要尽可能快地执行完毕,以确保该信号不被屏蔽.(为了避免一些竞态条件,信号在处理期间,系统不会再次出发它)太久.这里就采用一种常用的解决方案是:把信号的主要处理函数逻辑放到程序的主循环中,当信号处理函数被触发时,它只是简单地通知主循环程序接受到信号,并把信号值传递给主函数.主循环在根据接受到的信号值执行目标信号对应的处理逻辑代码.通常采用管道的方式来将"信号"传递给主循环.主程序采用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-统一事件源的更多相关文章

  1. 统一事件源epoll代码示例

    可以将信号注册进pipe管道的写端,通过对读端的监听,来实现统一事件源. #include <sys/types.h> #include <sys/socket.h> #inc ...

  2. study notes: high performance linux server programming

    1:linux网络API分为:socker地址API,socker基础API,网络信息API 1,socker地址API:包含IP地址和端口(ip, port).表示TCP通信的一端. 2,socke ...

  3. Linux高性能server编程——信号及应用

     信号 信号是由用户.系统或者进程发送给目标进程的信息.以通知目标进程某个状态的改变或系统异常. Linux信号可由例如以下条件产生: 对于前台进程.用户能够通过输入特殊的终端字符来给它发送信号. ...

  4. 第2章 Linux系统安装(2)_Linux系统分区及文件系统

    2. 系统分区 2.1 分区类型 (1)主分区:最多只能有4个 (2)扩展分区 ①最多只能有1个,主分区加扩展分区最多有4个. ②不能写入数据,只能包含逻辑分区 (3)逻辑分区 2.2 格式化(高级格 ...

  5. 13.linux中断处理程序

    linux中断处理程序 一.中断处理流程 在linux内核代码中进入entry-armv.S目录: linux统一的入口:__irq svc. 进入了统一的入口之后,程序跳到irq_handler标号 ...

  6. Linux爆新漏洞,长按回车键70秒即可获得root权限

    漏洞来源这个安全问题来源于Cryptsetup存在的一个漏洞(CVE-2016-4484).Cryptsetup是在Linux统一密钥设置(Linux Unified Key Setup, LUKS) ...

  7. linux概念之/proc与/sys

    http://blog.chinaunix.net/uid-1835494-id-3070465.html  proc/x:1/sched http://bbs.chinaunix.net/threa ...

  8. 【Linux】常用命令

    一.文件结构     /:       根目录 /bin:    系统所需要的那些命令位于此目录. /boot:   Linux的内核及引导系统程序所需要的文件目录,GRUB或LILO系统引导管理器也 ...

  9. 菜鸟必须知道的linux的文件目录结构

    Linux文件目录结 / 根目录,所有的目录.文件.设备都在/之下,/就是Linux文件系统的组织者,也是最上级的领导者. /bin bin就是二进制(binary)英文缩写.在一般的系统当中,你都可 ...

随机推荐

  1. jQuery异步加载数据添加事件

    几个月前在一个项目中涉及到树形栏,然后看了很多插件,觉得有点麻烦,于是自己写了一个,写着写着就出问题了. 当时项目是通过树形栏进行权限控制的,管理员可以对从数据库去的数据动态生成树形栏进行增删改查操作 ...

  2. 关于在transform下的子元素设置fixed无效的困惑

    最近的项目是要实现一个点击显示隐藏边栏的效果,而且需要边栏随着滚动而滚动. 思路简单,不就一个css的动画和一个position为fixed,搞定!但不想,设为fixed的子元素竟然无法fixed,这 ...

  3. smarty练习: 设置试题及打印试卷

    数据库表格:shiti, shititimu, timu, kemu, xuanxiang 根据科目设置一个可以添加试题的页面:(如下图) 具体的题目从数据库中查出并形成一张试卷的形式 考试试题设置: ...

  4. HTML标签区别

    一.div和span的区别 div是一个块级元素,可以包含段落,表格等内容,用于放置不同的内容.一般我们在网页通过div来布局定位网页中的每个区块. span是一个内联元素,没有实际意义,它的存在纯粹 ...

  5. canvas加载图像

    之前按照例子,加载图像,发现加载图像,加载不上去,代码也没有错误,经过几次的尝试:发现需要重复调用下drawImage才可以. <script type="text/javascrip ...

  6. android系统自带的Service原理与使用

    1. 说明 android的后台运行在很多service,它们在系统启动时被SystemServer开启,支持系统的正常工作,比如 MountService监听是否有SD卡安装及移除,Clipboar ...

  7. POJ 3384 Feng Shui

    http://poj.org/problem?id=3384 题意:给一个凸包,求往里面放两个圆(可重叠)的最大面积时的两个圆心坐标. 思路:先把凸包边往内推R,做半平面交,然后做旋转卡壳,此时得到最 ...

  8. T-SQL函数类别统计

  9. poj3708:函数式化简+高精度进制转换+同余方程组

    题目大意 给定一个函数 找出满足条件   等于 k 的最小的x m,k,d已知 其中 m,k 很大需要使用高精度存储 思路: 对 函数f(m)进行化简 ,令t=ceil( log(d,m) ) 可以得 ...

  10. 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 ...