select模型

在Linux中,我们可以使用select函数实现I/O端口的复用,传递给 select函数的参数会告诉内核:
      1) 我们所关心的文件描述符
      2) 对每个描述符,我们所关心的状态。(我们是要想从一个文件描述符中读或者写,还是关注一个描述符中是否出现异常)
      3) 我们要等待多长时间。(我们可以等待无限长的时间,等待固定的一段时间,或者根本就不等待)

从select函数返回后,内核告诉我们一下信息:
      1) 对我们的要求已经做好准备的描述符的个数
      2) 对于三种条件哪些描述符已经做好准备.(读,写,异常)

返回:做好准备的文件描述符的个数,超时为0,错误为 -1.

最后一个参数。它指明我们要等待的时间:
struct timeval{     
        long tv_sec;   /*秒 */
        long tv_usec;  /*微秒 */  
    }

   有三种情况:
    timeout == NULL  等待无限长的时间。等待可以被一个信号中断。当有一个描述符做好准备或者是捕获到一个信号时函数会返回。如果捕获到一个信号,

select函数将返回 -1,并将变量 erro设为 EINTR。
    timeout->tv_sec == 0 &&timeout->tv_usec == 0不等待,直接返回。加入描述符集的描述符都会被测试,并且返回满足要求的描述符的个数。这种方法通过轮

询,无阻塞地获得了多个文件描述符状态。
    timeout->tv_sec !=0 ||timeout->tv_usec!= 0 等待指定的时间。当有描述符符合条件或者超过超时时间的话,函数返回。在超时时间即将用完但又没有描述符

合条件的话,返回 0。对于第一种情况,等待也会被信号所中断。

函数返回:
 1) 当监视的相应的文件描述符集中满足条件时,比如说读文件描述符集中有数据到来时,内核(I/O)根据状态修改文件描述符集,并返回一个大于0的数。
 2) 当没有满足条件的文件描述符,且设置的timeval监控时间超时时,select函数会返回一个为0的值。
 3) 当select返回负值时,发生错误。

server

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

int main(int argc, char** argv)
{
    int listenfd1 = socket(AF_INET, SOCK_STREAM, 0);

    struct sockaddr_in serv_addr1;

    bzero((char*)&serv_addr1,sizeof(serv_addr1));
    serv_addr1.sin_family = AF_INET;
    serv_addr1.sin_port = htons(7778);
    serv_addr1.sin_addr.s_addr = INADDR_ANY;

    bind(listenfd1, (struct sockaddr*)&serv_addr1,sizeof(serv_addr1));
    listen(listenfd1,5);

    int maxfd = listenfd1;
    fd_set allset,rset;

    FD_ZERO(&allset);
    FD_SET(listenfd1,&allset);

    int clifd, clilen;
    struct sockaddr_in cli_addr;
    char buffer[256];
    for(;;) {
        rset = allset;
        select(maxfd + 1, &rset, NULL, NULL, NULL);

        //如果是listenfd1 获取消息
        if(FD_ISSET(listenfd1, &rset)) {
            clilen = sizeof(cli_addr);
            clifd = accept(listenfd1, (struct sockaddr *) &cli_addr, (socklen_t*)&clilen);

            bzero(buffer, 256);
            read(clifd, buffer, 255);
            printf("Listenfd1 Message is:%s\r\n", buffer);
        }
    }

    close(listenfd1);
}
View Code

client

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

int main(int argc, char* argv[])
{
    int socketfd;
    socketfd = socket(AF_INET,SOCK_STREAM,0);
    struct sockaddr_in serv_addr;
    bzero((char*)&serv_addr,sizeof(serv_addr));
    serv_addr.sin_family = AF_INET;
    serv_addr.sin_port = htons(7778);

    connect(socketfd,(struct sockaddr*)&serv_addr,sizeof(serv_addr));

    write(socketfd,"client_message",14);

    return 0;
}
View Code


 

posted on 2014-10-31 21:55  kangbry  阅读(158)  评论(0编辑  收藏  举报

导航