【问题标题】:C: Server/client socket program - Client error connectingC:服务器/客户端套接字程序 - 客户端错误连接
【发布时间】:2023-08-19 11:24:01
【问题描述】:

我是网络新手,正在尝试用 C 语言创建一个简单的客户端、服务器套接字程序,其中参数决定程序应该作为客户端还是服务器运行。我通过使用简单的 if 语句来做到这一点(如果给出了标志,则作为服务器运行,否则作为客户端运行),但我不确定如何测试它。我在一个终端(例如,在 localhost 和端口号 3000 上)运行我的代码,并将参数设置为服务器,然后打开另一个终端并运行带有参数的代码作为客户端(也在 localhost 和同一端口上)。

预期的结果是看到客户端提示用户输入消息(如果连接成功),并将该消息发送到服务器,服务器打印出消息,但是,我在客户端上没有得到提示输入消息。

(我从许多在线网站之一获得了服务器和客户端行为的代码,但它们将 client.c 和 server.c 分开,而我想将两者合并到一个 .c 程序中)

下面是我的代码,错误是由

触发的
if (connect(sockfd,(struct sockaddr *)&serv_addr,sizeof(serv_addr)) < 0)
       error("ERROR connecting");

在代码的客户端部分。

#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netdb.h>
#include <string.h>

void error(char *msg)
{
    perror(msg);
    exit(1);
}

int main(int argc, char *argv[])
{
 int sockfd, newsockfd, portno, clilen;
 char buffer[256];
 struct sockaddr_in serv_addr, cli_addr;
 int n, i, server = 0;  // 1 = server, 0 = client

 // check if server or client
 for (i = 0; i<argc; i++) {
   if (strcmp(argv[i], "-l") == 0)
      server = 1;
 }

 // client
 if (server == 0) {
   struct hostent *server;

   if (argc < 3) {
      fprintf(stderr,"usage %s hostname port\n", argv[0]);
      exit(0);
   }
   portno = atoi(argv[2]);
   sockfd = socket(AF_INET, SOCK_STREAM, 0);
   if (sockfd < 0)
       error("ERROR opening socket");
   server = gethostbyname(argv[1]);
   if (server == NULL) {
       fprintf(stderr,"FOUR*** ERROR, no such host\n");
       exit(0);
   }
   bzero((char *) &serv_addr, sizeof(serv_addr));
   serv_addr.sin_family = AF_INET;
   bcopy((char *)server->h_addr,
        (char *)&serv_addr.sin_addr.s_addr,
        server->h_length);
   serv_addr.sin_port = htons(portno);
   if (connect(sockfd,(struct sockaddr *)&serv_addr,sizeof(serv_addr)) < 0)
       error("ERROR connecting");
   printf("Please enter the message: ");
   bzero(buffer,256);
   fgets(buffer,255,stdin);
   n = write(sockfd,buffer,strlen(buffer));
   if (n < 0)
        error("ERROR writing to socket");
   bzero(buffer,256);
   n = read(sockfd,buffer,255);
   if (n < 0)
        error("ERROR reading from socket");
   printf("%s\n",buffer);
   return 0;
 }

 // server
 if (server == 1) {

   sockfd = socket(AF_INET, SOCK_STREAM, 0);
   if (sockfd < 0)
      error("ERROR opening socket");
   bzero((char *) &serv_addr, sizeof(serv_addr));
   portno = atoi(argv[1]);
   serv_addr.sin_family = AF_INET;
   serv_addr.sin_addr.s_addr = INADDR_ANY;
   serv_addr.sin_port = htons(portno);
   if (bind(sockfd, (struct sockaddr *) &serv_addr,
            sizeof(serv_addr)) < 0)
            error("ERROR on binding");
   listen(sockfd,5);
   clilen = sizeof(cli_addr);
   newsockfd = accept(sockfd, (struct sockaddr *) &cli_addr, &clilen);
   if (newsockfd < 0)
        error("ERROR on accept");
   bzero(buffer,256);
   n = read(newsockfd,buffer,255);
   if (n < 0) error("ERROR reading from socket");
   printf("Here is the message: %s\n",buffer);
   n = write(newsockfd,"I got your message",18);
   if (n < 0) error("ERROR writing to socket");
   return 0;
 }
}

这是确切的错误输出:

在一个终端窗口中,我首先将程序作为服务器运行:

$ ./socketz -l localhost 2003

然后在另一个终端窗口中,我作为客户端运行程序:

$ ./socketz localhost 2003
ERROR connecting: Connection refused

【问题讨论】:

  • 这是一个很长的程序,没有任何包含,所以不容易看,也不容易编译。请务必包含重现您的问题所需的一切。但是快速浏览一下就会知道argv[i] == "-l" 永远不是您想要的(即它会总是失败;并且您的编译器可能已经警告过您)。使用strcmp 比较字符串。

标签: sockets network-programming client-server


【解决方案1】:

您无法连接的原因是因为您的服务器进程没有在端口 2003 上侦听。特别是在这一行:

 serv_addr.sin_port = htons(portno);

portno 的值为零,这导致serv_addr.sin_port 的值也为零,accept() 将其解释为它应该只选择一个可用的 TCP 端口来绑定。

问题的根源在这里:

 portno = atoi(argv[1]);

...该行根据您在运行程序时输入的第一个参数为portno 分配一个值,但是您输入了这个:

 ./a.out -l localhost 2003

所以第一个参数是“-l”,它是一个非数字,所以它会导致 atoi() 返回 0。我认为你的意图是 portno = atoi(argv[3]);

【讨论】:

  • 您实际上并不需要服务器的主机名,所以./a.out -l 2003 就足够了,arg[2]
最近更新 更多