基于select模型的udp客户端实现超时机制

参考:http://www.cnblogs.com/chenshuyi/p/3539949.html

多路选择I/O — select模型

其思想在于使用一个集合,该集合中包含需要进行读写的fd,通过轮询这个集合,直到有一个fd可读写,才返回。与阻塞I/O不同的是,阻塞I/O仅使用了一次系统调用,就是对fd的读写,如果没有fd处于就绪状态,则进程一直阻塞,而多路选择I/O使用了两次系统调用,第一次是轮询并返回可读写fd数,第二次是对fd进行读写,阻塞只发生在轮询fd的过程。

select函数的原型(sys/select.h)

1 int select (int __nfds, fd_set *__restrict __readfds,
2          fd_set *__restrict __writefds,
3          fd_set *__restrict __exceptfds,
4          struct timeval *__restrict __timeout);

(1)__nfds

    需轮询的最大文件描述符数。如__nfds = 10,则轮询值为0~9的fd,单个进程中,最多可打开1024个fd,该值在sys/select.h中的FD_SETSIZE定义。用户可通过“ulimit -n”查看该值,通过打印/proc/sys/fs/file-max中的值查看系统可打开的最大fd数。

(2)__readfds,__writefds,__exceptfds

    分别代表用户关心的可读、可写、异常的fd,这三个参数的数据类型是fd_set *,这是一组文件描述符的集合,使用一个位来代表一个fd。

fd_set位向量操作函数包括

1 #define FD_SET(fd, fdsetp)   __FD_SET (fd, fdsetp)   //将指定的fd置1
2 #define FD_CLR(fd, fdsetp)   __FD_CLR (fd, fdsetp)   //将指定的fd清0
3 #define FD_ISSET(fd, fdsetp) __FD_ISSET (fd, fdsetp) //测试fd状态,如被置1,返回非0,否则返回0
4 #define FD_ZERO(fdsetp)      __FD_ZERO (fdsetp)      //将所有的fd清0

(3)__timeout

    timeout的数据类型是timeval结构体。通过填充该结构体,设置超时时间,精确到微妙级,如果该参数设置为NULL,则一直等待,直到有fd可读写。如果tv_sec和tv_usec都设置为0,则轮询完指定fd后,立即返回。

1 struct timeval
2 {
3     __time_t tv_sec;          /* Seconds.  */
4     __suseconds_t tv_usec;    /* Microseconds.  */
5 };

(4)select有三种返回值

-1  :出错

0   :如果设置了超时,在指定时间内没有fd可读写,则返回0,可在此指定相应的超时处理操作。

>0 :返回可读写的fd数

可屏蔽信号的select — pselect

从原型上看,pselect函数与select函数的区别在于设置超时的结构体不同,以及多了个用于屏蔽信号的参数。如果__sigmask设置为NULL,则与select一样。

1 int pselect (int __nfds, fd_set *__restrict __readfds,
2             fd_set *__restrict __writefds,
3             fd_set *__restrict __exceptfds,
4             const struct timespec *__restrict __timeout,
5             const __sigset_t *__restrict __sigmask);

(1)timespec结构体的定义如下,它精确到纳秒级。

1 struct timespec
2 {
3     __time_t tv_sec;         /* Seconds.  */
4     long int tv_nsec;        /* Nanoseconds.  */
5 };

(2)__sigmask实际上是信号的位向量。数据类型是sigset_t,定义如下

1 /* A `sigset_t‘ has a bit for each signal.  */
2
3 # define _SIGSET_NWORDS    (1024 / (8 * sizeof (unsigned long int)))
4 typedef struct
5 {
6     unsigned long int __val[_SIGSET_NWORDS];
7 } __sigset_t;

测试代码:

//client
#include <Winsock2.h>
#include <stdio.h>
#pragma comment (lib,"Ws2_32.lib")

#define MAX_LINE 80
#define PORT 8000
typedef int socklen_t;

int main(int argc, char *argv[])
{
    struct sockaddr_in   sin, cin;
    socklen_t addr_len;

    int sockfd, maxfdp;
    char *msg = "client";
    char buf[MAX_LINE];

    fd_set fds;
    struct timeval timeout;
    int interval = 3;

    memset(&sin, 0,sizeof(sin));

    sin.sin_family = AF_INET;
    //inet_pton(AF_INET, "127.0.0.1", &sin.sin_addr);
    sin.sin_addr.S_un.S_addr = inet_addr("127.0.0.1");
    sin.sin_port = htons(PORT);

    if ((sockfd = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP)) < 0) {
        perror("fail to create socket.\n");
        exit(1);
    }

    addr_len = sizeof(cin);

    while (1) {

        timeout.tv_usec = 0;
        timeout.tv_sec = interval;

        FD_ZERO(&fds);
        FD_SET(sockfd, &fds);

        maxfdp = sockfd + 1;

        sendto(sockfd, msg, strlen(msg) + 1, 0, (struct sockaddr*)&sin, sizeof(sin));

        switch (select(maxfdp, &fds, NULL, NULL, &timeout)) {
        case -1:
            perror("error");
            exit(-1);
            break;

        case 0:
            printf("timeout.\n");
            break;

        default:
            if (FD_ISSET(sockfd, &fds)) {

                if (recvfrom(sockfd, buf, MAX_LINE, 0, (struct sockaddr *)&cin, &addr_len) < 0) {
                    perror("fail to receive.\n");
                    exit(1);
                }
                else {
                    printf("receive from server: %s.\n", buf);

                    if (closesocket(sockfd) < 0) {
                        perror("fail to close.\n");
                        exit(1);
                    }
                    return 0;
                }
            }
            break;
        }

    }
    return 0;
}
//Server#pragma comment (lib,"ws2_32.lib")
#include <Winsock2.h>
#include <stdio.h>
#define MAX_LINE 80
#define PORT 8000

int main(int argc, char *argv[])
{
    struct sockaddr_in sin, cin;
    int addr_len;

    int sockfd, n;
    char *msg = "server";
    char buf[MAX_LINE];

    memset(&sin,0, sizeof(sin));

    sin.sin_family = AF_INET;
    sin.sin_addr.s_addr = htons(INADDR_ANY);
    sin.sin_port = htons(PORT);

    if ((sockfd = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP)) < 0) {
        perror("fail to create socket.\n");
        exit(1);
    }

    n = bind(sockfd, (struct sockaddr*)&sin, sizeof(sin));
    if (n < 0) {
        perror("fail to bind.\n");
        exit(1);
    }

    while (1) {
        addr_len = sizeof(cin);

        recvfrom(sockfd, buf, MAX_LINE, 0, (struct sockaddr*)&cin, &addr_len);

        sendto(sockfd, msg, strlen(msg) + 1, 0, (struct sockaddr*)&cin, addr_len);
    }
}

测试:链接失败。。。

时间: 2024-10-06 00:07:23

基于select模型的udp客户端实现超时机制的相关文章

基于Select模型的混乱聊天室v1.0

最近在无聊完成了一个简单的基于select模型的匿名聊天室程序,均使用C++开发 服务器工作原理: 每接收一条客户端的信息,就将遍历所有的socket,并将该信息发给所有的客户端. 客户端使用两条线程,一个是接收服务端信息的线程,一个是等待阻塞输入的线程,获得输入时,将输入发送到服务器. 项目源码:https://github.com/coderguang/Chat 版本为v2.0的release. 其中ComLib也在github上 服务器核心代码: int main(int argc,cha

c#基于事件模型的UDP通讯框架(适用于网络包编解码)

之前写过一篇关于c#udp分包发送的文章 这篇文章里面介绍的方法是一种实现,可是存在一个缺点就是一个对象序列化后会增大非常多.不利于在网络中的传输. 我们在网络中的传输是须要尽可能的减小传送的数据包的大小.于是我參考了网上一些资料和一些开源的项目(http://www.fishlee.net/)这个上面的那个开源的飞鸽传输的框架. 事实上也就是把要传送的数据依照某种规定放在一个byte数组中,然后接收到后依照对应的格式把数据解析出来,为了减小数据还使用了GZipStream的压缩,之前出的问题就

Linux统系统开发12 Socket API编程3 TCP状态转换 多路IO高并发select poll epoll udp组播 线程池

[本文谢绝转载原文来自http://990487026.blog.51cto.com] Linux统系统开发12 Socket API编程3 TCP状态转换 多路IO高并发select  poll  epoll udp组播 线程池 TCP 11种状态理解: 1,客户端正常发起关闭请求 2,客户端与服务端同时发起关闭请求 3,FIN_WAIT1直接转变TIME_WAIT 4,客户端接收来自服务器的关闭连接请求 多路IO转接服务器: select模型 poll模型 epoll模型 udp组播模型 线

基于EPOLL模型的局域网聊天室和Echo服务器

一.EPOLL的优点 在Linux中,select/poll/epoll是I/O多路复用的三种方式,epoll是Linux系统上独有的高效率I/O多路复用方式,区别于select/poll.先说select/poll的缺点,以体现epoll的优点. select: (1)可监听的socket受到限制,在32位的系统中,默认最大值为1024. (2)采用轮询方式,当要监听的sock数量很大时,效率低. (3)随着要监听socket数据的增加,要维护一个存放大量fd的数据结构,系统开销太大. pol

socket编程:多路复用之select模型

系统提供select函数来实现多路复用输入/输出模型. select函数让我们的程序监视多个文件描述符的状态变化.程序会停在select这里等待,直到被监视的文件描述符中有一个或多个发生了状态变化 函数原型如下: 返回值:   成功返回就绪描述符的个数,超过timeout时间且没有任何事件发生返回0,失败返回-1 参数解释: nfds:    被监视的文件描述符中值最大描述符值加1(描述符是从0开始的,描述符0.1.2...nfds-1均将被测试) 下面三个参数readset.writeset和

服务器select模型

多路转接I/O服务器中的一种:select服务器,该模型的服务器是将文件描述符放入队列中保存并监听,以轮询的机制去监听这些文件描述符,当相对应的文件描述符有读请求.写情况或异常发生时,对应的位将发生变化.select模型需要对所有监听的套接字实行轮询监听处理,当需要监听的套接字过多时,就可能出现响应不及时等问题,从而降低了服务器性能. 下面是服务器的实现(服务器将客户端发送过来的数据全部转换为大写) #include <stdio.h> #include <fcntl.h> #in

select模型的原理、优点、缺点

关于I/O多路复用: I/O多路复用(又被称为“事件驱动”),首先要理解的是,操作系统为你提供了一个功能,当你的某个socket可读或者可写的时候,它可以给你一 个通知.这样当配合非阻塞的socket使用时,只有当系统通知我哪个描述符可读了,我才去执行read操作,可以保证每次read都能读到有效数据而不 做纯返回-1和EAGAIN的无用功.写操作类似.操作系统的这个功能通过select/poll/epoll之类的系统调用来实现,这些函数都可以同时 监视多个描述符的读写就绪状况,这样,**多个描

Winsock IO模型之select模型

之所以称其为select模型是因为它主要是使用select函数来管理I/O的.这个模型的设计源于UNIX系统,目的是允许那些想要避免在套接字调用上阻塞的应用程序有能力管理多个套接字. int select( int nfds,                                                 // 忽略,仅是为了与Berkeley套接字兼容 fd_set* readfds,                                  // 指向一个套接字集合,

socket select模型

由于socket recv()方法是堵塞式的,当多个客户端连接服务器时,其中一个socket的recv调用时,会产生堵塞,使其他连接不能继续. 如果想改变这种一直等下去的焦急状态,可以多线程来实现(不再等待,同时去recv,同时阻塞),每个socket连接使用一个线程,这样效率十分低下,根本不可能应对负荷较大的情况(是啊,占用各种资源,电脑啊,你耗不起). 这时候我们便可以采取select模型.select允许进程指示内核等待多个事件中的任何一个发生,并仅在有一个或多个事件发生或经历一段指定时间