【问题标题】:Client-server text file transfer using UDP & sockets in C++, server creates a blank file在 C++ 中使用 UDP 和套接字进行客户端-服务器文本文件传输,服务器创建一个空白文件
【发布时间】:2013-05-07 01:32:18
【问题描述】:

首先,我想指出我不是一个优秀的程序员,所以请耐心等待。程序中的逻辑如下:客户端向服务器发送一个文本文件,服务器以不同的名称保存它。简而言之,与这个主题非常相似: “File transfer server/client using socket”除了我使用的是不同的协议。我想我已经成功地发送了文件,但似乎服务器在创建一个空白文件后陷入了循环/没有在其中写入任何内容。

这是客户端的代码:

#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <netdb.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h> /* memset() */
#include <sys/time.h> /* select() */ 
#include <errno.h>
#include <sys/stat.h>
#include <fcntl.h>  


#define REMOTE_SERVER_PORT 1500
#define MAX_MSG 100
#define PLIK    "/home/aatami/Documents/programowanie/serwer/plik.txt"
#define LENGTH 512
int main(int argc, char *argv[]) {

  int sd, rc, i,fd;
  unsigned int cliLen;
  struct sockaddr_in cliAddr, remoteServAddr;
  struct hostent *h;
  char  buf[LENGTH];
  /* check command line args */
  if(argc<3) {
    printf("usage : %s <server> <data1> ... <dataN> \n", argv[0]);
    exit(1);
  }

  /* get server IP address (no check if input is IP address or DNS name */
  h = gethostbyname(argv[1]);
  if(h==NULL) {
    printf("%s: unknown host '%s' \n", argv[0], argv[1]);
    exit(1);
  }

  printf("%s: sending data to '%s' (IP : %s) \n", argv[0], h->h_name,
     inet_ntoa(*(struct in_addr *)h->h_addr_list[0]));

  remoteServAddr.sin_family = h->h_addrtype;
  memcpy((char *) &remoteServAddr.sin_addr.s_addr, 
     h->h_addr_list[0], h->h_length);
  remoteServAddr.sin_port = htons(REMOTE_SERVER_PORT);

  /* socket creation */
  sd = socket(AF_INET,SOCK_DGRAM,0);  /* port 0 - system gets the first free one */ 
  if(sd<0) {
    printf("%s: cannot open socket \n",argv[0]);
    exit(1);
  }

  /* bind any port */
  cliAddr.sin_family = AF_INET;
  cliAddr.sin_addr.s_addr = htonl(INADDR_ANY);
  cliAddr.sin_port = htons(0);

  rc = bind(sd, (struct sockaddr *) &cliAddr, sizeof(cliAddr));
  if(rc<0) {
    printf("%s: cannot bind port\n", argv[0]);
    exit(1);
  }


  /* send data */
  for(i=2;i<argc;i++) {
    rc = sendto(sd, argv[i], strlen(argv[i])+1, 0, 
        (struct sockaddr *) &remoteServAddr, 
        sizeof(remoteServAddr));

    if(rc<0) {
      printf("%s: cannot send data %d \n",argv[0],i-1);
      close(sd);
      exit(1);
    }
    /* send text file */


        char sdbuf[LENGTH]; 
        printf("[Client] Sending %s to the Server... ", PLIK);
        FILE *fs = fopen(PLIK, "r");
        if(fs == NULL)
        {
            printf("ERROR: File %s not found.\n", PLIK);
            exit(1);
        }

        bzero(sdbuf, LENGTH); 
        int fs_block_sz; 
        while((fs_block_sz = fread(sdbuf, sizeof(char), LENGTH, fs)) > 0)
        {
            if(send(sd, sdbuf, fs_block_sz, 0) < 0)
            {
                fprintf(stderr, "ERROR: Failed to send file %s. (errno = %d)\n", PLIK,           errno);
                break;
            }
            bzero(sdbuf, LENGTH);
        }
        printf("Ok File %s from Client was Sent!\n", PLIK);


  }

  return 1;

}

以及服务器的代码:

#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <netdb.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h> /* close() */
#include <string.h> /* memset() */
#include <errno.h>
#define LENGTH 512
#define LOCAL_SERVER_PORT 1500
#define MAX_MSG 100
#define PLIKSERV    "/home/aatami/Documents/programowanie/serwer/plikserv.txt"
int main(int argc, char *argv[]) { /* licznik argumentow, tablica argumentow */

  int sd, rc,nsockfd;
  unsigned int n,cliLen;
  struct sockaddr_in cliAddr, servAddr;
  char msg[MAX_MSG];
   char  buf[512],sbuf[LENGTH];
  /* socket creation */
  sd=socket(AF_INET, SOCK_DGRAM, 0);
  if(sd<0) {
    printf("%s: cannot open socket \n",argv[0]);
    exit(1);
  }

  /* bind local server port */
  servAddr.sin_family = AF_INET;
  servAddr.sin_addr.s_addr = htonl(INADDR_ANY);
  servAddr.sin_port = htons(LOCAL_SERVER_PORT);
  rc = bind (sd, (struct sockaddr *) &servAddr,sizeof(servAddr));
  if(rc<0) {
    printf("%s: cannot bind port number %d \n", 
       argv[0], LOCAL_SERVER_PORT);
    exit(1);
  }

  printf("%s: waiting for data on port UDP %u\n", 
       argv[0],LOCAL_SERVER_PORT);



  /* server infinite loop */
  while(1) {

    /* init buffer */
    memset(msg,0x0,MAX_MSG);


    /* receive message */
    cliLen = sizeof(cliAddr);
    n = recvfrom(sd, msg, MAX_MSG, 0, 
         (struct sockaddr *) &cliAddr, &cliLen);

    if(n<0) {
      printf("%s: cannot receive data \n",argv[0]);
      continue;
    }

    /* print received message */
    printf("%s: from %s:UDP%u : %s \n", 
       argv[0],inet_ntoa(cliAddr.sin_addr),
       ntohs(cliAddr.sin_port),msg);

         unsigned int sin_size = sizeof(struct sockaddr_in);




        FILE *fr = fopen(PLIKSERV, "a");
        if(fr == NULL)
            printf("File %s Cannot be opened file on server.\n", PLIKSERV);
        else
        {
            bzero(sbuf, LENGTH); 
            int fr_block_sz = 0;
            while((fr_block_sz = recvfrom(sd, buf, LENGTH, 0,(struct sockaddr *) &cliAddr, &cliLen)) > 0) 
            {
                int write_sz = fwrite(buf, sizeof(char), fr_block_sz, fr);
                if(write_sz < fr_block_sz)
                {
                    perror("File write failed on server.\n");
                }
                bzero(buf, LENGTH);
                if (fr_block_sz == 0 || fr_block_sz != 512) 
                {
                    break;
                }
            }
            if(fr_block_sz < 0)
            {
                if (errno == EAGAIN)
                {
                    printf("recv() timed out.\n");
                }
                else
                {
                    fprintf(stderr, "recv() failed due to errno = %d\n", errno);
                    exit(1);
                    }
                }
            printf("Ok received from client!\n");
            fclose(fr); 
        }

  }/* end of server infinite loop */

return 0;

}

非常感谢任何帮助!

【问题讨论】:

  • 我看到的一件事是,在recvfrom 中,您正在阅读sbuf,但在fwrite 中,您正在从buf 中写入。
  • 我都改成buf了,结果还是一样。
  • 这段代码毫无意义。您正在编写 UDP 代码,就好像它是 TCP 代码一样。例如,您的服务器读取循环在客户端永远不会发送的零长度数据包上终止。您还在测试 EAGAIN,它永远不会在您正在使用的阻塞模式下发生。您没有为 UDP 数据报丢失或重复或乱序到达做任何准备。您的客户端正在发送 argc-2 数据包,但您的服务器正在接收一个。您的客户端在未连接的 UDP 套接字上使用 send(),这将产生 ENOTCONN 错误。扔掉它,重新开始。

标签: c++ sockets udp file-transfer


【解决方案1】:

看到您的代码后,我建议您尝试从客户端向服务器发送“Hello world\n”,看看您是否收到了它。一旦您在客户端和服务器之间建立连接,然后尝试扩展代码以进行文件传输。

但请记住关于 UDP 的这一点:

"UDP 使用简单的传输模型和最少的协议机制。它没有握手对话,因此将底层网络协议的任何不可靠性暴露给用户的程序。由于这通常是不可靠媒体上的 IP,所以没有交付、订购或重复保护的保证。如果在网络接口级别需要纠错功能,应用程序可以使用为此目的设计的传输控制协议 (TCP)。"

当您启动客户端时,它会尽可能快地发送所有数据包,可能是数据包丢失或混乱甚至在服务器站点重复。我还建议等待数据包确认正确接收。我的意思是,当服务器收到一个数据包时,它会发送一个数据包以向客户端确认它。这样,可以进行对话(对话),并且可以令人满意地完成传输。

【讨论】:

  • 其实我已经用TCP重写了,现在我只是固执地用UDP写。这里有一种方法可以检查是否已经实现了某些工作-服务器从客户端接收到一条消息,该消息在终端中打印出来,并且工作正常。如果我一次又一次地尝试发送(消息和文件),最终的文本文件会损坏 - 它不是空的,而是有一些 1 和 0 和 /。是因为重复/混乱/等包吗?
  • 恐怕是这样。可能有 10 种情况传输正常,有时不正常,这是因为重复/无序/丢失数据包; UDP 不是为此目的而设计的。无论采用哪种方式,UDP 或 TCP,您都需要创建一个协议来指示您将收到什么。
  • 如何为 UDP 做到这一点?这对于新手程序员来说是可行的吗,还是我最好离开 TCP 传输?非常感谢您的建议!
猜你喜欢
  • 1970-01-01
  • 2012-07-13
  • 1970-01-01
  • 1970-01-01
  • 2012-06-30
  • 1970-01-01
  • 1970-01-01
  • 2012-12-01
  • 1970-01-01
相关资源
最近更新 更多