select 设置发送超时发送注意事项

//设置发送超时
你只发送, 并发送足够多的数据以填满发送缓冲区, 接收端一直不接收.
发送端一量满发送缓冲区就会阻塞, 若是你设置了发送超时, 超时到了它就会返回发送超时了.网络

在send(),recv()过程当中有时因为网络情况等缘由,收发不能预期进行,而设置收发超时控制:
在Linux下须要注意的是时间的控制结构是struct timeval而并非某一整型数,如下是来自于网上一篇文章中的摘录,它是这样写的:
int nNetTimeout=1000;//1秒,
//设置发送超时
setsockopt(socket,SOL_SOCKET,SO_SNDTIMEO,(char *)&nNetTimeout,sizeof(int));
//设置接收超时
setsockopt(socket,SOL_SOCKET,SO_RCVTIMEO,(char *)&nNetTimeout,sizeof(int));
 
这样作在Linux环境下是不会产生效果的,须以下定义:struct timeval timeout = {3,0}; 
// 设置发送超时
setsockopt(socket,SOL_SOCKET,SO_SNDTIMEO,(char *)&timeout,sizeof(struct timeval));
// 设置接收超时
setsockopt(socket,SOL_SOCKET,SO_RCVTIMEO,(char *)&timeout,sizeof(struct timeval));
有两点注意就是:
1)recv ()的第四个参数需为MSG_WAITALL,在阻塞模式下不等到指定数目的数据不会返回,除非超时时间到。还要注意的是只要设置了接收超时,在没有MSG_WAITALL时也是有效的。说到底超时就是不让你的程序老在那儿等,到必定时间进行一次返回而已。
2)即便等待超时时间值未到,但对方已经关闭了socket, 则此时recv()会当即返回,并收到多少数据返回多少数据。
 

在进行程序开发时,有时候须要阻塞,但同时又须要有超时功能,这时候select()函数就能很好的知足咱们的要求:并发

但用这进行测试时有一个地方是须要注意的,即select()的第五个参数timeval *timeout的问题。设置好timeout的始值后,若是只对select()调用一次,是没有任何问题的,但一旦屡次调用,你就会发现怎么好像我设置的timeout值只有第一次有用呢,之后select()老是一刻也不等就返回了呢?socket

经过查看man和本身测试,原来select()函数内部是会不断更新timeout的值的,以查看超时时间还剩多少。那么第一次调用以后,timeout的值就被更新至0了,之后无论你再调用它多少次,select()都会当即返回了,这就是为何会出如今屡次调用时,select()只有第一次时凑效的原因了,哈哈,之后再使用它时,要尤为注意这一点了。但它的兄弟函数pselect()没有它的这个问题,它不会在内部在时间值进行更新的。函数

socket在每次执行select的时候都得要从新对time进行赋值 防止select修改了time参数 对下一次的判断形成干扰
for (i = 0 ; i < waittimeout ; i++)
    {
        FD_SET(fd , &fdr);
        sTime.tv_sec = 1;
        sTime.tv_usec = 0;        iRet = select(iMax , &fdr , NULL , NULL , &sTime);
        if (iRet == -1)
        {
            LOG_TRACE(&gLogger, "select -1");
            return -1;
        }
        else if (iRet > 0)
        {
            LOG_TRACE(&gLogger, "wait %ds sock readalbe, select=%d", i, iRet);
            return 0 ;
        }
    }
相关文章
相关标签/搜索