C: Server/client 套接字程序 - 客户端错误连接

C: Server/client socket program - Client error connecting

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

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

(我从许多在线网站之一获得了服务器和客户端行为的代码,但它们将 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;
 }
}

这是准确的错误输出:

在一个终端window中,我先运行程序作为服务器:

$ ./socketz -l localhost 2003

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

$ ./socketz localhost 2003
ERROR connecting: Connection refused

您无法连接的原因是您的服务器进程未在端口 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() 到 return 0。我认为你的意图是 portno = atoi(argv[3]); .