网络专题二:信号驱动IO

信号驱动IO是由操作系统回调到用户态的一种网络IO模型:

首先我们允许套接口进行信号驱动 I/O, 并安装一个信号处理函数,进程继续运行并不阻
塞。当数据准备好时,进程会收到一个 SIGIO 信号,可以在信号处理函数中调用 I/O 操作函
数处理数据。当数据报准备好读取时,内核就为该进程产生一个 SIGIO 信号。我们随后既可
以在信号处理函数中调用 read 读取数据报,并通知主循环数据已准备好待处理,也可以立
即通知主循环,让它来读取数据报。无论如何处理 SIGIO 信号,这种模型的优势在于等待数
据报到达 ( 第一阶段 ) 期间,进程可以继续执行,不被阻塞。免去了 select 的阻塞与轮询,当
有活跃套接字时,由注册的 handler 处理。

#include<stdio.h>
#include<sys/socket.h>
#include<sys/types.h>
#include<netinet/in.h>
#include<fcntl.h>
#include<signal.h>
#include<string.h>
#include<unistd.h>

int sockfd = 0;
void do_sigio(int sig) {
    //从内核里面回调出来的
    char buffer[256]={0};
    struct sockaddr_in clientaddr;
    int len = sizeof(struct sockaddr_in);
    int rlen = recvfrom(sockfd,buffer,256,0,(struct sockaddr*)&clientaddr,(socklen_t*)&len);
    int slen = sendto(sockfd,buffer,rlen, 0 ,(struct sockaddr*)&clientaddr,len);
    
}

int main() {
    
    struct sigaction sigio_action;
    sigio_action.sa_flags = 0;
    sigio_action.sa_handler = do_sigio; //信号处理函数
    
    sigaction(SIGIO,&sigio_action,NULL);//绑定信号和信号处理函数
    sockfd = socket(AF_INET,SOCK_DGRAM,0);
    
    struct sockaddr_in serv_addr;
    memset(&serv_addr,0,sizeof(serv_addr));
    serv_addr.sin_family = AF_INET;
    serv_addr.sin_port = htons(6016);
    serv_addr.sin_addr.s_addr = INADDR_ANY;
    
    bind(sockfd,(struct sockaddr*)&serv_addr,sizeof(serv_addr));
    
    //设置这个fd属于哪个进程
    fcntl(sockfd,F_SETOWN,getpid());
    //获取这个fd的状态
    int flags = fcntl(sockfd,F_GETFL,0);
    flags |= O_ASYNC | O_NONBLOCK;
    //再设置这个FD非阻塞异步
    flags = fcntl(sockfd,F_SETFL,flags);
    
    while(1) sleep(1);
    return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值