Unix网络编程之给阻塞式函数设定超时限定

1.给read函数设定超时时间。

代码:

int read_timeout(int fd,unsigned int wait_seconds)
{
        int ret = 0;
        if(wait_seconds > 0){
                fd_set set;
                struct timeval timeout;
                FD_ZERO(&set);
                FD_SET(fd,&set);
                timeout.tv_sec = wait_seconds;
                timeout.tv_usec = 0;
                do{
                        ret = select(fd+1,&set,NULL,NULL,&timeout);
                }while(ret < 0 && errno == EINTR);

                if(ret == 0){
                        ret = -1;
                        errno = ETIMEDOUT;
                }
                else if(ret == 1)
                        ret = 0;
        }

        return ret;
}

2.给write函数设定超时时间。

代码:

int write_timeout(int fd,unsigned int wait_seconds){
        int ret = 0;
        if(wait_seconds > 0){
                fd_set set;
                struct timeval timeout;
                FD_ZERO(&set);
                FD_SET(fd,&set);
                timeout.tv_sec = wait_seconds;
                timeout.tv_usec = 0;
                do{
                        ret = select (fd+1,NULL,&set,&timeout);
                }while(ret <0 && errno == EINTR);

                if(ret == 0){
                        ret = -1;
                        errno == ETIMEDOUT;
                }
                else if(ret == 1){
                        ret = 0;
                }
        }
        return ret;
}

3.给accept函数设定超时限定。

代码:

int accept_timeout(int fd,struct sockaddr_in *addr,unsigned int wait_seconds){
        int ret = 0;
        socklen_t addrlen = sizeof(struct sockaddr_in);
        if(wait_seconds > 0){
                fd_set set;
                struct timeval timeout;
                FD_ZERO(&set);
                FD_SET(fd,&set);
                timeout.tv_sec = wait_seconds;
                timeout.tv_usec = 0;
                do{
                        ret = select(fd+1,&set,NULL,NULL,&timeout);
                }while(ret < 0 && errno == EINTR);

                if(ret == -1)
                        return -1;
                else if(ret == 0){
                        errno = ETIMEDOUT;
                }
        }
        if(addr != NULL){
                accept(fd,(struct sockaddr*)addr,&addrlen);
        }else
            ret = accept(fd,NULL,NULL);
        if(ret == -1)
                ERR_EXIT("accept");
        return ret;
}

4.给connect 函数设定超时限定。

代码:

int connect_timeout(int fd,struct sockaddr_in *addr,unsigned int wait_seconds){
        int ret;
        socklen_t addrlen = sizeof(struct sockaddr_in);
        if(wait_seconds > 0){
                activate_nonblock(fd);
        }
        ret = connect(fd,(struct sockaddr*)addr,addrlen);
        if(ret < 0&& errno == EINPROGRESS){
                fd_set set;
                struct timeval timeout;
                FD_ZERO(&set);
                FD_SET(fd,&set);
                timeout.tv_sec = wait_seconds;
                timeout.tv_usec = 0;
                do{
                        ret = select(fd+1,NULL,&set,NULL,&timeout);
                }while(ret < 0 && errno == EINTR);

                if(ret == 0){
                        ret = -1;
                        errno = ETIMEDOUT;
                }else if(ret < 0)
                        return -1;
                 else if(ret ==1)
                {
                        int err;
                        socklen_t socklen = sizeof(err);
                        int sockoptret = getsockopt(fd,SOL_SOCKET,SO_ERROR,&err,&socklen);
                        if(err == 0){
                                ret = 0;

                        }else{
                                errno = err;
                                ret = -1;
                        }
                }

        }
        if(wait_seconds > 0)
        {
                deactivate_nonblock(fd);
        }
        return ret;
}

5.给一个fd设置阻塞还是非阻塞的函数:

(1)设定fd为非阻塞

void deactivate_nonblock(int fd){

        int ret;
        int flags = fcntl(fd,F_GETFL);
        if(flags ==-1)
                ERR_EXIT("fcntl");
        flags &= ~O_NONBLOCK;
        ret = fcntl(fd,F_SETFL,flags);
        if(ret == -1)
                ERR_EXIT("fcntl");
}
(2)设定fd为阻塞:

void activate_nonblock(int fd){
        int ret;
        int flags = fcntl(fd,F_GETFL);
        if(flags == -1){
                ERR_EXIT("fcntl");
        }
        flags |= O_NONBLOCK;
        ret = fcntl(fd,F_SETFL,flags);
        if(ret == -1){
                ERR_EXIT("fcntl");
        }
}



  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
UNIX网络编程之道 第一卷》是一本经典的计算机网络编程书籍。该书的目的是教授读者如何使用UNIX操作系统进行网络编程,提供了丰富的理论知识和实际应用示例。 该书内容由两部分组成,第一部分介绍了UNIX网络编程的基础知识,包括套接字、地址结构、标准I/O等内容。作者通过简洁、明确的语言,深入浅出地讲解了这些关键概念。读者可以从中了解到UNIX网络编程的基本原理和技术,为后续章节的学习打下坚实的基础。 第二部分重点介绍了网络编程的高级特性和实际应用。这些内容包括主机和服务的名称解析、TCP和UDP的编程、多进程和多线程的编程、I/O复用、高级的套接字选项等。作者详细讲解了这些概念的原理和用法,并提供了大量的示例代码和实战案例。通过学习这些内容,读者可以掌握在UNIX环境下进行网络编程的各种技巧和方法,为实际项目的开发提供了有力的支持。 此外,该书还介绍了一些UNIX网络编程中常见的问题和解决方法,如错误处理、并发编程、安全问题等。作者结合自身丰富的实践经验,提供了一些建议和技巧,帮助读者更好地完成网络编程任务。 总之,《UNIX网络编程之道 第一卷》是一本非常优秀的网络编程专著。它深入浅出地介绍了UNIX网络编程的基础知识和实际应用,为读者提供了宝贵的学习资料和参考指南。无论是初学者还是有一定经验的开发者,都可以从中获得丰富的知识和技巧,提升自己在网络编程领域的能力。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值