读书人

socket的select模式// if(fdSocket.fd

发布时间: 2013-08-06 16:47:25 作者: rapoo

socket的select模式// if(fdSocket.fd_array[i] == sListen)的else之后{里怎么没有accept}?
本帖最后由 oyljerry 于 2013-07-28 23:33:01 编辑


USHORT nPort = 4567; // 此服务器监听的端口号

// 创建监听套节字
SOCKET sListen = ::socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
sockaddr_in sin;
sin.sin_family = AF_INET;
sin.sin_port = htons(nPort);
sin.sin_addr.S_un.S_addr = INADDR_ANY;
// 绑定套节字到本地机器
if(::bind(sListen, (sockaddr*)&sin, sizeof(sin)) == SOCKET_ERROR)
{
printf(" Failed bind() \n");
return -1;
}
// 进入监听模式
::listen(sListen, 5);
2.


// select模型处理过程
// 1)初始化一个套节字集合fdSocket,添加监听套节字句柄到这个集合
fd_set fdSocket; // 所有可用套节字集合
FD_ZERO(&fdSocket);
FD_SET(sListen, &fdSocket);
3.


while(TRUE)
{
// 2)将fdSocket集合的一个拷贝fdRead传递给select函数,
// 当有事件发生时,select函数移除fdRead集合中没有未决I/O操作的套节字句柄,然后返回。
fd_set fdRead = fdSocket;
int nRet = ::select(0, &fdRead, NULL, NULL, NULL);
if(nRet > 0)
{
// 3)通过将原来fdSocket集合与select处理过的fdRead集合比较,
// 确定都有哪些套节字有未决I/O,并进一步处理这些I/O。
for(int i=0; i<(int)fdSocket.fd_count; i++)
{
if(FD_ISSET(fdSocket.fd_array[i], &fdRead))
{
if(fdSocket.fd_array[i] == sListen) // (1)监听套节字接收到新连接
{
if(fdSocket.fd_count < FD_SETSIZE)
{
sockaddr_in addrRemote;
int nAddrLen = sizeof(addrRemote);


SOCKET sNew = ::accept(sListen, (SOCKADDR*)&addrRemote, &nAddrLen);

FD_SET(sNew, &fdSocket);
printf("接收到连接(%s)\n", ::inet_ntoa(addrRemote.sin_addr));
}
else
{
printf(" Too much connections! \n");
continue;
}
}
else
{
char szText[256];
int nRecv = ::recv(fdSocket.fd_array[i], szText, strlen(szText), 0);
if(nRecv > 0) // (2)可读
{
szText[nRecv] = '\0';
printf("接收到数据:%s \n", szText);
}


else // (3)连接关闭、重启或者中断
{
::closesocket(fdSocket.fd_array[i]);

printf("关闭\n");
FD_CLR(fdSocket.fd_array[i], &fdSocket);
}
}
}
}
}
else
{
printf(" Failed select() \n");
break;
}
}


[解决办法]
引用:
新连接的读写事件 为啥不需要accept进行接收啊?
Quote: 引用:

不是有accept吗? 对于sListen来说 可读事件就是来连接了 所以只要fdSocket.fd_count < FD_SETSIZE

就accept 然后把新连接FD_SET(sNew, &fdSocket);

这个 if(fdSocket.fd_array[i] == sListen) 对应的else说明不是侦听事件 是新连接的读写事件 所以去

recv或者send(当然你这代码里没处理写事件)


回的时候忘看发帖人了 当我什么都没说...


[解决办法]


//下面else代码中怎么没有accept呢?
else
{
char szText[256];
int nRecv = ::recv(fdSocket.fd_array[i], szText, strlen(szText), 0);
if(nRecv > 0) // (2)可读 {
szText[nRecv] = '\0';
printf("接收到数据:%s \n", szText); } else // (3)连接关闭、重启或者中断 {
::closesocket(fdSocket.fd_array[i]);
printf("关闭\n");
FD_CLR(fdSocket.fd_array[i], &fdSocket);
}
}
\


引用:
SOCKET sNew = ::accept(sListen, (SOCKADDR*)&addrRemote, &nAddrLen);


怎么没有.
他只是先判断了sock是自己监听的还是接受的.才能操作.socket的select模式// if(fdSocket.fd_array[i] == sListen)的else之后{里如何没有accept}
话说这代码 貌似是那个xx网络编程书的代码.我也啃过.

读书人网 >VC/MFC

热点推荐