【问题标题】:c windows connect() fails. error 10049c windows connect() 失败。错误 10049
【发布时间】:2011-02-20 17:49:44
【问题描述】:

以下两段代码编译,但我在客户端收到connect() failed 错误。 (使用 MinGW 编译)。

客户代码:

// thanks to cs.baylor.edu/~donahoo/practical/CSockets/code/TCPEchoClientWS.c

#include <stdio.h>
#include <winsock.h>
#include <stdlib.h>

#define RCVBUFSIZE 32 // size of receive buffer

void DieWithError(char *errorMessage);

int main(int argc, char* argv[])
{
  int sock;
  struct sockaddr_in echoServAddr;
  unsigned short echoServPort;
  char *servIP;
  char *echoString;
  char echoBuffer[RCVBUFSIZE];
  int echoStringLen;
  int bytesRcvd, totalBytesRcvd;
  WSAData wsaData;

  if((argc < 3) || (argc > 4)){
    fprintf(stderr, "Usage: %s <Sever IP> <Echo Word> [<Echo Port>]\n", argv[0]);
    exit(1);
  }

  if (argc==4)
    echoServPort = atoi(argv[3]); // use given port if any
  else
    echoServPort = 7; // echo is well-known port for echo service

  if(WSAStartup(MAKEWORD(2, 0), &wsaData) != 0){ // load winsock 2.0 dll
    fprintf(stderr, "WSAStartup() failed");
    exit(1);
  }

  // create reliable, stream socket using tcp
  if((sock=socket(PF_INET, SOCK_STREAM, IPPROTO_TCP)) < 0)
    DieWithError("socket() failed");

  // construct the server address structure
  memset(&echoServAddr, 0, sizeof(echoServAddr));
  echoServAddr.sin_family = AF_INET;
  echoServAddr.sin_addr.s_addr = inet_addr(servIP); // server IP address
  echoServAddr.sin_port = htons(echoServPort);
  // establish connection to the echo server
  if(connect(sock, (struct sockaddr*)&echoServAddr, sizeof(echoServAddr)) < 0)
    DieWithError("connect() failed");

  echoStringLen = strlen(echoString); // determine input length

  // send the string, includeing the null terminator to the server
  if(send(sock, echoString, echoStringLen, 0)!= echoStringLen)
    DieWithError("send() sent a different number of bytes than expected");

  totalBytesRcvd = 0;
  printf("Received: "); // setup to print the echoed string
  while(totalBytesRcvd < echoStringLen){
    // receive up to the buffer size (minus 1 to leave space for a null terminator) bytes from the sender
    if(bytesRcvd = recv(sock, echoBuffer, RCVBUFSIZE-1, 0) <= 0)
      DieWithError("recv() failed or connection closed prematurely");
    totalBytesRcvd += bytesRcvd; // keep tally of total bytes
    echoBuffer[bytesRcvd] = '\0';
    printf("%s", echoBuffer); // print the echo buffer
  }

  printf("\n");
  closesocket(sock);
  WSACleanup(); 

  exit(0);
}

void DieWithError(char *errorMessage)
{
  fprintf(stderr, "%s: %d\n", errorMessage, WSAGetLastError());
  exit(1);
}

服务器代码:

// thanks cs.baylor.edu/~donahoo/practical/CSockets/code/TCPEchoServerWS.c

#include <stdio.h>
#include <winsock.h>
#include <stdlib.h>

#define MAXPENDING 5 // maximum outstanding connection requests
#define RCVBUFSIZE 1000

void DieWithError(char *errorMessage);
void HandleTCPClient(int clntSocket); // tcp client handling function

int main(int argc, char **argv)
{
  int serverSock;
  int clientSock;
  struct sockaddr_in echoServerAddr;
  struct sockaddr_in echoClientAddr;
  unsigned short echoServerPort;
  int clientLen; // length of client address data structure
  WSAData wsaData;

  if (argc!=2){
    fprintf(stderr, "Usage: %s <Server Port>\n", argv[0]);
    exit(1);
  }

  echoServerPort = atoi(argv[1]);

  if(WSAStartup(MAKEWORD(2, 0), &wsaData)!=0){
    fprintf(stderr, "WSAStartup() failed");
    exit(1);
  }

  // create socket for incoming connections
  if((serverSock=socket(PF_INET, SOCK_STREAM, IPPROTO_TCP))<0)
    DieWithError("socket() failed");

  // construct local address structure
  memset(&echoServerAddr, 0, sizeof(echoServerAddr));
  echoServerAddr.sin_family = AF_INET;
  echoServerAddr.sin_addr.s_addr = htonl(INADDR_ANY); // any incoming interface
  echoServerAddr.sin_port = htons(echoServerPort); // local port

  // bind to the local address
  if(bind(serverSock, 
      (struct sockaddr*)&echoServerAddr, 
      sizeof(echoServerAddr)
      )<0)
    DieWithError("bind() failed");

  // mark the socket so it will listen for incoming connections
  if(listen(serverSock, MAXPENDING)<0)
    DieWithError("listen() failed");

  for (;;){ // run forever
    // set the size of the in-out parameter
    clientLen = sizeof(echoClientAddr);

    // wait for a client to connect
    if((clientSock = accept(serverSock, (struct sockaddr*)&echoClientAddr,
                &clientLen)) < 0)
      DieWithError("accept() failed");

    // clientSock is connected to a client

    printf("Handling client %s\n", inet_ntoa(echoClientAddr.sin_addr));

    HandleTCPClient(clientSock);
  }

  // NOT REACHED
}

void DieWithError(char *errorMessage)
{
  fprintf(stderr, "%s: %d\n", errorMessage, WSAGetLastError());
  exit(1);
}

void HandleTCPClient(int clientSocket)
{
  char echoBuffer[RCVBUFSIZE]; // buffer for echostring
  int recvMsgSize; // size of received message

  // receive message from client
  if((recvMsgSize = recv(clientSocket, echoBuffer, RCVBUFSIZE, 0) <0))
    DieWithError("recv() failed");

  // send received string and receive again until end of transmission
  while(recvMsgSize > 0){
    // echo message back to client
    if(send(clientSocket, echoBuffer, recvMsgSize, 0)!=recvMsgSize)
      DieWithError("send() failed");

    // see if there's more data to receive
    if((recvMsgSize = recv(clientSocket, echoBuffer, RCVBUFSIZE, 0)) <0)
      DieWithError("recv() failed");
  }

  closesocket(clientSocket); // close client socket
}

我该如何解决这个问题?

【问题讨论】:

  • @Joshua Moore 这是作业吗?看起来代码很好,但也许你的本地主机有一些防火墙规则?几周前我完成了这项工作(简单的客户端/服务器),但它是针对 linux 的......我徘徊,我的代码是用 MinGW 编译的吗?! :)
  • @mosg 几个月前我把 linux 版本作为家庭作业。几天前,我偶然发现了这些文件,它们是为 windows 修改的 linux 代码。 cs.baylor.edu/~donahoo/practical/CSockets/winsock.html
  • @Joshua Moore 好的 :) 很好的作业!简单的 codeka 是对的。

标签: c windows networking tcp


【解决方案1】:

你没有初始化servIP:

servIP = argv[1];

我认为您确实需要熟悉您的调试器,因为使用调试器您应该能够在几秒钟内发现这种错误。

【讨论】:

  • 我没有使用调试器。老实说,我真的不知道如何使用。在学校里,我们被简要介绍了 GDB。好学吗?
  • 我添加了 servIP = argv[1];和 echoString = argv[2] 在第一个 if 块之后。 Connect() 现在可以工作,但我收到一个 recv() 错误:recv() 失败或连接过早关闭:0
【解决方案2】:

问题出在客户端。

  1. servIP 未分配给正确的值。您可以使用 servIP = argv[1]; 在加载 winsock2 dll 之前。

  2. recv(...) 返回 0 属于正常情况,表示连接已关闭。

【讨论】:

  • 这是否意味着 recv 行应该读为if (recv(...) &lt;0) DieWithError() 而不是if(recv(...) &lt;=0) DieWithError()?此外,该程序应该打印接收到的字符串。而是打印错误消息
  • 是的,它应该是 if (recv(...)
猜你喜欢
  • 2019-09-01
  • 1970-01-01
  • 1970-01-01
  • 2013-03-28
  • 2011-08-27
  • 1970-01-01
  • 2013-09-01
  • 2016-08-12
  • 2014-02-16
相关资源
最近更新 更多