2015-05-13 42 views
0

我目前正在爲項目編寫一個tcp服務器類和一個ClientHandle類。 是否有可能,服務器類只調用fd_set上的select僅包含偵聽器來檢查新客戶端是否連接?使用選擇只在偵聽套接字?

內部的句柄我想做類似的套接字來檢查是否有新的請求,或者如果它是可寫之前發送數據。

這樣的事情?

int TCPServer::start() 
{ 
    WSADATA wsaData; 
    int iResult; 

    // Initialize Winsock 
    iResult = WSAStartup(MAKEWORD(2, 2), &wsaData); 
    if (iResult != 0) 
    { 
     printf("WSAStartup failed: %d\n", iResult); 
     return 1; 
    } 

    struct addrinfo *result = nullptr, *ptr = nullptr, hints; 

    ZeroMemory(&hints, sizeof (hints)); 
    hints.ai_family = AF_INET; 
    hints.ai_socktype = SOCK_STREAM; 
    hints.ai_protocol = IPPROTO_TCP; 
    hints.ai_flags = AI_PASSIVE; 

    // Resolve the local address and port to be used by the server 
    iResult = getaddrinfo(nullptr, DEFAULT_PORT, &hints, &result); 
    if (iResult != 0) 
    { 
     printf("getaddrinfo failed: %d\n", iResult); 
     WSACleanup(); 
     return 1; 
    } 

    // Create a SOCKET for the server to listen for client connections 
    //create socket to listen for clients 
    m_listensocket = INVALID_SOCKET; 
    m_listensocket = socket(result->ai_family, result->ai_socktype, 
          result->ai_protocol); 
    if (m_listensocket == INVALID_SOCKET) 
    { 
     printf("Error at socket(): %ld\n", WSAGetLastError()); 
     freeaddrinfo(result); 
     WSACleanup(); 
     return 1; 
    } 

    // Setup the TCP listening socket 
    iResult = bind(m_listensocket, result->ai_addr, (int)result->ai_addrlen); 
    if (iResult == SOCKET_ERROR) 
    { 
     printf("bind failed with error: %d\n", WSAGetLastError()); 
     freeaddrinfo(result); 
     closesocket(m_listensocket); 
     WSACleanup(); 
     return 1; 
    } 

    //free the result 
    freeaddrinfo(result); 

    //start listening 
    if (listen(m_listensocket, SOMAXCONN) == SOCKET_ERROR) 
    { 
     printf("Listen failed with error: %ld\n", WSAGetLastError()); 
     closesocket(m_listensocket); 
     WSACleanup(); 
     return 1; 
    } 
    //clear master set 
    FD_ZERO(&m_set); 
    //add the listen socket to the set 
    FD_SET(m_listensocket, &m_set); 
    return 0; 
} 
int TCPServer::acceptClient() 
{ 
    auto temp_set = m_set; 

    //setup the timeout to 100ms 
    struct timeval tv; 
    tv.tv_sec = 0; 
    tv.tv_usec = 100; 

    if (select(1, &temp_set, nullptr, nullptr, &tv) == -1) 
    { 
     //failed to select 
     perror("fail in select method!"); 
     exit(4); 
    } 
    if (FD_ISSET(m_listensocket, &temp_set)) 
    { 
     //we can accept a new 
     SOCKET ClientSocket; 
     ClientSocket = INVALID_SOCKET; 
     // Accept a client socket 
     struct sockaddr_storage their_addr; //stored address 
     int sin_size = sizeof their_addr; 
     ClientSocket = accept(m_listensocket, 
           reinterpret_cast<struct sockaddr *>(&their_addr), 
           &sin_size); 
     if (ClientSocket == INVALID_SOCKET) 
     { 
      printf("accept failed: %d\n", WSAGetLastError()); 
      closesocket(m_listensocket); 
      WSACleanup(); 
      return 1; 
     } 
     auto handle = new ClientHandle(ClientSocket); 
     //now do whatever you want 
    } 
} 

回答

1

是否有可能,服務器類調用Select只包含listensocket以檢查是否有新的客戶端連接一個FD_SET?

是的,就是這樣。

在句柄的內部我想做類似的套接字來檢查是否有新的請求,或者如果它是可寫的之前發送數據。

是的,就是這樣。

+0

非常感謝您對這個可能無用的問題的簡短而明確的答案。感到困惑,因爲我需要把它放到集合中。但是,如果應該一起檢查多個連接,那麼肯定有用。 – BennX