1

我有以下基本的 C++ 代码,用于使用 linux 上的套接字通过 UDP 传输文件。问题是,并不是所有的数据包都被传递(比如只收到 345kb 中的 8 个字节),我知道 UDP 不可靠,但这与我的代码有关还是仅仅因为 UDP 的不可靠性?

客户端

sockfd=socket(AF_INET,SOCK_DGRAM,0);

bzero(&servaddr,sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_addr.s_addr=inet_addr("192.168.0.152");
servaddr.sin_port=htons(32000);
ifstream myfile("345kb.doc", ios::in);
int i=0;
char* memblock;
double size;
if(myfile.is_open()) {
    myfile.seekg(0,ios::end);
    size=(double)myfile.tellg();
    memblock=new char[1024];
    myfile.seekg(0,ios::beg);
    double filesize=size;
    cout<<filesize;
    int n;
    int ab=(int)filesize%1024;
    while(filesize>1024){
        myfile.read(memblock,1024);
        cout<<"\nREAD:"<<strlen(memblock)<<"\n"<<"SENT:";
        n=sendto(sockfd,memblock,strlen(memblock),0,(struct sockaddr *)&servaddr,sizeof(servaddr));
        filesize=filesize-1024;
        cout<<n<<"\n";
    }
    myfile.read(memblock,1024);
    n=sendto(sockfd,memblock,strlen(memblock),0,(struct sockaddr *)&servaddr,sizeof(servaddr));
    std::cout<<"HI"<<endl;
} else {
    printf("\nFILE NOT OPENED \n");
}

服务器端

int sockfd,n;
struct sockaddr_in servaddr,cliaddr;
socklen_t len;
char mesg[1024];

sockfd=socket(AF_INET,SOCK_DGRAM,0);

bzero(&servaddr,sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_addr.s_addr=htonl(INADDR_ANY);
servaddr.sin_port=htons(32000);
bind(sockfd,(struct sockaddr *)&servaddr,sizeof(servaddr));

//for (;;)
//{
    ofstream myfile;
    myfile.open("345kbsssss.doc",ios::out|ios::app);
    if(myfile.is_open()) {
        long i=0;
        len = sizeof(cliaddr);
        while((n = recvfrom(sockfd,mesg,1024,0,(struct sockaddr *)&cliaddr,&len))>0){
            std::cout<<"HI\n";
            //n=recvfrom(sockfd,mesg,256,0,(struct sockaddr *)&cliaddr,&len);
            myfile.write(mesg,n);
        }
    }
    else printf("\nFILE NOT CREATED\n");
//}
4

1 回答 1

3

两者都是真的:

  • 如果不执行检查、重新排序和重新传输,UDP 并不意味着用于文件传输。与其通过 UDP 实现所有这些,不如只切换到 TCP。除了资源有限的嵌入式系统之外。

  • 在您strlen用于获取块长度的代码中,由于文件(很可能)是二进制文件,这会产生不正确的长度值,并且您不会传输整个文件数据。strlen对以零结尾的字符串进行操作,并且只计算数据中的第一个 0 字节。

要获得更正确的结果,最简单的方法是替换strlen(memblock)min(1024,filesize). 这不是一个完美的解决方案,因为它不检查到底有多少字节被读取myfile.read

于 2013-07-15T10:26:42.967 回答