为什么这样用select,筛选式使用FD_ISSET 不行?

本文探讨了在使用socket编程时,为何在筛选套接字集合中使用FD_ISSET函数是必要的,而非简单的筛选式操作。通过示例代码展示了标准的for循环结构在检查活动文件描述符集中的重要性。

 

#include <stdio.h>
#include <sys/socket.h>
#include <stdlib.h>
#include <netinet/in.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <errno.h>

#define N 64

typedef struct sockaddr SA;

int main(int argc,char *argv[])
{
	int listenfd,connfd[16],i = 0,maxfd;
	int n;
	struct sockaddr_in  myaddr,peeraddr,connaddr;
	socklen_t peerlen = sizeof(connaddr),connlen = peerlen; 

	char buf[N];

	fd_set readfd,fdsetbak;

	bzero(connfd,sizeof(connfd));

	//	if(argc < 3)
	//	{
	//		printf("Usage: %s <IP> <Port>\n",argv[0]);
	//		exit(-1);
	//	}

	if((listenfd = socket(PF_INET,SOCK_STREAM,0)) < 0)
	{
		perror("socket error");
		exit(-1);
	}

	bzero(&myaddr, sizeof(myaddr));
	myaddr.sin_family = PF_INET;
	myaddr.sin_port = htons(atoi("9090"));
	myaddr.sin_addr.s_addr = inet_addr("192.168.0.103");
	//	myaddr.sin_port = htons(atoi(argv[2]));
	//	myaddr.sin_addr.s_addr = inet_addr(argv[1]);

	if(bind(listenfd, (SA *)&myaddr, sizeof(myaddr)) < 0)
	{
		perror("fail to bind");
		exit(-1);
	}

	if(listen(listenfd, 16) < 0)
	{
		perror("fail to listen");
		exit(-1);
	}

	FD_ZERO(&readfd);
	FD_SET(listenfd,&readfd);
	maxfd = listenfd;
	fdsetbak = readfd;

	while(1)
	{
		readfd = fdsetbak;
		if(select(maxfd+1, &readfd, NULL,NULL,NULL) < 0)
		{
			perror("fail to select");
			exit(-3);
		}

		if(FD_ISSET(listenfd,&readfd))
		{
			i = 0;
			while(connfd[i])
			{
				i++;
				if(i>=16) break;	// chao chu 16;
			}

			if((connfd[i] =accept(listenfd,(SA *)&peeraddr,&peerlen)) < 0)
			{
				perror("fail to accept");
				i = 0;
				exit(-2);
			}
			FD_SET(connfd[i],&readfd);
			fdsetbak = readfd;
			maxfd = maxfd>connfd[i] ? maxfd:connfd[i];
			getsockname(connfd[i],(SA *)&connaddr,&connlen);
			printf("new connect from [%s:%d]to[%s:%d]\n",inet_ntoa(peeraddr.sin_addr),ntohs(peeraddr.sin_port),inet_ntoa(connaddr.sin_addr),ntohs(connaddr.sin_port));

		}
		else
		{
			for(i = 0;i<16;i++)
			{
				if(connfd[i])
				{
					if(FD_ISSET(connfd[i],&readfd))
					{
						if((n = recv(connfd[i],buf,N,0)) <=0)
						{
							getpeername(connfd[i],(SA *)&peeraddr,&peerlen);
							printf("[%s:%d] closed\n",inet_ntoa(peeraddr.sin_addr),ntohs(peeraddr.sin_port));
							FD_CLR(connfd[i],&fdsetbak );
							readfd = fdsetbak; 
							close(connfd[i]);
							connfd[i] = 0;
						}
						else
						{
							printf("send to %d\n",connfd[i]);
							send(connfd[i],buf,n,0);
						}

					}
				}
			}
		}
	}


	return 0;
}


 

筛选式的使用FD_ISSET不行吗?

非要

for(i=0;i<maxfd+1;i++)

{

          FD_ISSET();

         {

          }

}

 

才行吗?

 

for (rid = 0; rid < MAX_RADIO_NUM; rid++) { if (TRUE == scanError[rid]) { RA_ERROR("continue if %s scanning request failed.\r\n", iflist[rid]); continue; } /* init paramter */ memset(&cmdFd, 0, sizeof(cmdFd)); memset(&setOfSock, 0, sizeof(setOfSock)); if(0 != unix_sock_initSrv(UNIX_SOCK_ID_MANAGER_AP_SCAN, &cmdFd, 10)) { RA_ERROR("init command socket failed"); continue; } starScanTime = util_getUptime(UTIL_UNIT_S); currentTime = starScanTime; /*等待返回的扫描数据,设置最大等待时间为30s*/ RA_DEBUG("wait to rcev data"); while(currentTime > 0 && (currentTime - starScanTime) < 30) { unix_sock_groupClear(&setOfSock); unix_sock_groupAddTo(cmdFd.fd, &setOfSock); if(0 < unix_sock_groupWait(&setOfSock, AP_SCAN_SELECT_TIMEOUT)) { if(unix_sock_isSet(cmdFd.fd, &setOfSock)) { /*接收扫描获取的数据, 获取后退出循环*/ if(0 != _apscanRecvCmd(cmdFd.fd, &bss_msg, &scanResults)) { RA_DEBUG("msg recv ERROR, still waiting."); break; } else { RA_DEBUG("msg recv success."); break; } } } else { RA_DEBUG("no msg recv, still waiting."); currentTime = util_getUptime(UTIL_UNIT_S); continue; } currentTime = util_getUptime(UTIL_UNIT_S); } unix_sock_close(&cmdFd); if (scanResults->count == 0) { RA_DEBUG("scan result count is 0\n"); continue; }else if (scanResults->version != WL_BSS_INFO_VERSION) { RA_DEBUG("bss_info_version %d is not support\n", scanResults->version); continue; } bssInfo = (wl_bss_info_t*)scanResults->bss_info; for(i = 0; i < scanResults->count && n < AP_SCAN_MAX_NUM; i++) { memcpy(pRaTable->tRAEntry[n].ssid, bssInfo->SSID, MAX_SSID_LEN); memcpy(pRaTable->tRAEntry[n].bssid, (unsigned char*)(&bssInfo->BSSID), ETHER_ADDR_LEN); channel = wf_chspec_ctlchan(bssInfo->chanspec); pRaTable->tRAEntry[n].freq = CHANNUM_TO_MHZ(channel); pRaTable->tRAEntry[n].bcnIntval = bssInfo->beacon_period; pRaTable->tRAEntry[n].signal = bssInfo->RSSI; pRaTable->tRAEntry[n].keyType = 0; pRaTable->tRAEntry[n].bPSK = 0; raParseWPARSNIes((UINT8 *)(((UINT8 *)bssInfo) + bssInfo->ie_offset), bssInfo->ie_length, &(pRaTable->tRAEntry[n].keyType), &(pRaTable->tRAEntry[n].bPSK)); /*process next bss info*/ n++; bssInfo = (wl_bss_info_t*)((char *)bssInfo + bssInfo->length); } RA_DEBUG("rid:%u, scan num = %u\n", rid, i); } 分析上述代码,为什么在打印了msg recv success.后没有打印数据了
11-06
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值