首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 开发语言 > VC/MFC >

socket的select模式// if(fdSocket.fd_array[i] == sListen)的else之后{里如何没有accept}

2013-08-06 
socket的select模式// if(fdSocket.fd_array[i] sListen)的else之后{里怎么没有accept}?本帖最后由 oyl

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网络编程书的代码.我也啃过.

热点排行