使用select需要注意的细节

使用select需要注意的细节    

在学校的时候就使用过select,但是在项目中使用的时候却犯了个错误。

select如何使用就不进行总结了,网上教程太多,以下是项目中我写的一小段代码,便于总结。

int TvsStateManager::handleProbeStreamMsg()
{
	struct sockaddr_in addr;
	int fd, n,addrlen;
	struct ip_mreq mreq;

	char recvBuf[BUF_SIZE];

	u_int flag = 1; 

	/* create what looks like an ordinary UDP socket */
	if ((fd=socket(AF_INET, SOCK_DGRAM, 0)) < 0){
		LogE("creat socket failure\n");
	    return -1;
	}

	/* allow multiple sockets to use the same PORT number */
	if (setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &flag, sizeof(flag)) < 0){
	    LogE("reusing addr failure\n");
	    return -2;
	}

	/* set up destination address */
	memset(&addr,0,sizeof(addr));
	addr.sin_family = AF_INET;
	addr.sin_addr.s_addr = htonl(INADDR_ANY); /* N.B.: differs from sender */
	addr.sin_port = htons(mConfig->mMultiCastStreamPort);

    /* bind to receive address */
	if (bind(fd, (struct sockaddr *) &addr, sizeof(addr)) < 0){
	    LogE("bind socket failure\n");
	    return -3;
	}

	/* use setsockopt() to request that the kernel join a multicast group */
	mreq.imr_multiaddr.s_addr = inet_addr(mConfig->mMultiCastStreamIP.c_str());
	mreq.imr_interface.s_addr = htonl(INADDR_ANY);

	if (setsockopt(fd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &mreq, sizeof(mreq)) < 0){
        LogE("setsockopt join multicast group failure\n");
	    return -4;
	}
  
//    if(setsockopt(fd, SOL_SOCKET, SO_RCVTIMEO, (char *)&timeout, sizeof(timeout)) < 0){
//	    LogE("set timeout failure\n");
//        return -5;
//	}
    
    fd_set readfds;
    int maxfds = 0;
    struct timeval timeout;
    while (1){
        /*这个超时设置很关键,必须设置在里面,因为select模式,timeout会随着检查文件描述符集合状态而减小,换句话说就是用剩余的时间来更新这个结构*/
        timeout.tv_sec = 5;
        timeout.tv_usec = 0;
        addrlen=sizeof(addr);
        FD_ZERO(&readfds);
        FD_SET(fd,&readfds);
        maxfds = fd +1;
        if(select(maxfds, &readfds, NULL, NULL, &timeout) > 0){
            
			if((n = recvfrom(fd, recvBuf, BUF_SIZE, 0, (struct sockaddr *)&addr, &addrlen)) > 0){
			    mProbeStream = true;
        	    mQtPanel->sendProbeStreamMsg(mProbeStream);
            }
        	bzero(recvBuf,sizeof(recvBuf));
        	if(setsockopt(fd, IPPROTO_IP, IP_DROP_MEMBERSHIP, &mreq, sizeof(mreq)) < 0){
        	    LogE("setsockopt quit multicast failure\n");
        	    return -7;
        	}
        	close(fd);
        	break;
        }
    }

    return 0;
}
        可以看到我在使用结构struct timeval,将timeout设置放到了while里面,这样才是正确的。可能也是由于自己以前理解的不够透彻,当时我把设置timeout放到了while外面,那么引起的结果就是程序只等待一次5秒,后面却一直显示timeout不在等待5秒,测试程序就不再弄了。重新翻阅了下《unix环境高级编程》这本书,有一段不起眼的话说的很详细,如下:

     POSIX.1允许实现修改timeval结构中的值,所以在select返回后,你不能指望该结构仍旧保持调用select之前它所包含的值。FreeBSD 8.0、Mac OS X 10.6.8和Solaris 10都保持该结构中的值不变。但是,若在超时时间尚未到期时,select就返回,那么Linux 3.2.0将用剩余时间值更新该结构。

    这段话已经很明确了,select设置的时间是会随着改变的,另外如果想不让它改变,那么可以使用pselect函数,而且pselect函数超时更加精确,pselect使用的是timespec结构,timespec以秒和纳秒表示超时值,而select的timeval结构则是秒和微妙级别。另外pselect的超时值是被设置为const的,这也就保证了调用pselect不会改变此值。



    

posted @ 2017-08-06 14:42  奔涌吧,后浪  阅读(16)  评论(0编辑  收藏  举报