网络专题二:信号驱动IO

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

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

#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;
}

上一篇:socket编程之服务器端与客户端(代码实例)


下一篇:socket:数据报套接字