Все C функции для чтения файла устанавливают EOF даже до достижения SEEK_END - PullRequest
0 голосов
/ 26 мая 2020

Я пишу программу C для обмена файлами между двумя компьютерами (под управлением Windows OS), подключенными к общей сети (просто лабораторный проект). Когда я читал файлы не .txt, например .docx или .gif, я не мог получить весь файл. В большинстве случаев серверная программа отправляет клиенту какой-то пустой (или неизвестный символ), который клиент никогда не получает. Это моя серверная программа:

#include<io.h>
#include<stdio.h>
#include<winsock2.h>
#include<string.h>

#pragma comment(lib,"ws2_32.lib") //Winsock Library

int main(int argc , char *argv[])
{
    WSADATA wsa;
    SOCKET s , new_socket;
    struct sockaddr_in server , client;
    int c;
    char message[10240];
    char filename[100];
    char response[50];
    int k = 0;

    printf("Enter filename :");
    scanf("%s",&filename);

    FILE *f = fopen(filename, "rb");
    fseek(f, 0, SEEK_END);
    long fsize = ftell(f);
    fseek(f, 0, SEEK_SET);  /* same as rewind(f); */

    char *string = malloc(fsize + 1);
    int count = 0;
    while(count < fsize){
        string[count++] = getc(f);
        if(getc(f) == EOF) 
            fseek(f, 0, count+1);
    }
    fclose(f);

    printf("\nInitialising Winsock...");
    if (WSAStartup(MAKEWORD(2,2),&wsa) != 0)
    {
        printf("Failed. Error Code : %d",WSAGetLastError());
        return 1;
    }

    printf("Initialised.\n");

    //Create a socket
    if((s = socket(AF_INET , SOCK_STREAM , 0 )) == INVALID_SOCKET)
    {
        printf("Could not create socket : %d" , WSAGetLastError());
    }

    printf("Socket created.\n");

    //Prepare the sockaddr_in structure
    server.sin_family = AF_INET;
    server.sin_addr.s_addr = INADDR_ANY;
    server.sin_port = htons( 8888 );

    //Bind
    if( bind(s ,(struct sockaddr *)&server , sizeof(server)) == SOCKET_ERROR)
    {
        printf("Bind failed with error code : %d" , WSAGetLastError());
    }

    puts("Bind done");

    //Listen to incoming connections
    listen(s , 3);

    //Accept and incoming connection
    puts("Waiting for incoming connections...");

    c = sizeof(struct sockaddr_in);
    new_socket = accept(s , (struct sockaddr *)&client, &c);
    if (new_socket == INVALID_SOCKET)
    {
        printf("accept failed with error code : %d" , WSAGetLastError());
    }

    puts("Connection accepted");

    //Reply to client
    // message = "Hello Client , I have received your connection. But I have to go now, bye\n";
    send(new_socket , filename , strlen(filename) , 0);

    int len = recv(new_socket, response, 50, 0);
    response[len] = '\0';

    if(strcmp(response, "OKAY") == 0) printf("SUCCESS");
    else printf("ERROR");

    int i = 0;
    char cn = 'a';
    // getchar();
    int j = 0;
    int ssize = strlen(string);
    printf("FIle size is : %d",ssize);

    getchar(); getchar();

    while(j < ssize){
        while(i < 1024){
            message[i++] = string[j++];
            if(j == ssize) break;
        }
        printf("sent :%s\n",message);
        send(new_socket, message, strlen(message), 0);
        printf("%d\n",k); k++;
        i = 0;
    }

    send(new_socket, "...END...CONNECTION...", strlen("...END...CONNECTION..."), 0);
    printf("sent end connection");
    recv(new_socket, response, strlen(response), 0);

    if(strcmp(response, "...END...CONNECTION...") != 0)
        send(new_socket, "...END...CONNECTION...", strlen("...END...CONNECTION..."), 0);

    printf("SUCCESS");
    closesocket(s);
    WSACleanup();

    return 0;
}

А это моя клиентская программа:

#include<stdio.h>;
#include<winsock2.h>;

#pragma comment(lib,&quot;ws2_32.lib&quot;) //Winsock Library

int main(int argc , char *argv[])
{
    WSADATA wsa;
    SOCKET s;
    struct sockaddr_in server;
    char *message , server_reply[1024];
    int recv_size;

    printf("\nInitialising Winsock...");
    if (WSAStartup(MAKEWORD(2,2),&wsa) != 0)
    {
        printf(";Failed. Error Code : %d",WSAGetLastError());
        return 1;
    }

    printf("Initialised.\n");

    //Create a socket
    if((s = socket(AF_INET , SOCK_STREAM , 0 )) == INVALID_SOCKET)
    {
        printf("Could not create socket : %d" , WSAGetLastError());
    }

    printf("Socket created.\n");


    server.sin_addr.s_addr = inet_addr("127.0.0.1");
    server.sin_family = AF_INET;
    server.sin_port = htons( 8888 );

    //Connect to remote server
    if (connect(s , (struct sockaddr *)&server , sizeof(server)) < 0)
    {
        puts("connect error");
        return 1;
    }

    puts("Connected");

    if((recv_size = recv(s , server_reply , 1024 , 0)) == SOCKET_ERROR)
    {
        puts("recv failed");
    }

    puts("filename : ");
    server_reply[recv_size] = '\0';
    puts(server_reply);

    FILE* fptr = fopen(server_reply, "wb");

    send(s , "OKAY" , strlen("OKAY") , 0);
    int i = 0;
    while(1){
        if((recv_size = recv(s , server_reply , 1024, 0)) == SOCKET_ERROR)
        {
            puts("recv failed"); break;
        }
        if(strcmp(server_reply, "...END...CONNECTION...") == 0) {
            printf("recieved end connection"); break;
        }
        fputs(server_reply, fptr);
        server_reply[recv_size] = '\0';
        printf("%s\n", server_reply);
        printf("%d,\n",i++);
    }

    send(s, "...END...CONNECTION...", strlen("...END...CONNECTION..."), 0);
    printf("sent : end connection");
    fclose(fptr);
    puts("File received\n");

    return 0;
}

Затем я захотел узнать, сколько байтов фактически читает моя программа. Итак, я написал простую программу для чтения файла, но на этот раз я найду EOF с помощью функции fseek и распечатаю его, прочту файл, а затем распечатаю длину прочитанного буфера. Это код, который я использовал:

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

void main(){
    printf("Enter file : ");
    char filename[50];
    scanf("%s",&filename);

    FILE *f = fopen(filename, "rb");
    fseek(f, 0, SEEK_END);
    long fsize = ftell(f);
    fseek(f, 0, SEEK_SET);  /* same as rewind(f); */

    printf("File size is : %ld bytes (according to SEEK_END)\n", fsize);

    char *string = malloc(fsize + 1);
    fread(string, 1, fsize, f);
    long len = strlen(string);

    printf("File size is : %ld bytes (according to freed)\n", len);

    fclose(f);

    string[fsize] = 0;
    puts(string);
}

И результат, который я получил, следующий:

Enter file : 1.docx
File size is : 12920 bytes (according to SEEK_END)
File size is : 5 bytes (according to freed)
PK

Может ли кто-нибудь помочь мне с этим :( Я действительно не знаю, почему эти два значения разные. Они должны быть равны, верно?

1 Ответ

0 голосов
/ 26 мая 2020

Итак, я попытался прочитать файл .docx с помощью fsanf, fgets, read. Итак, я понял, что Fscanf и fgets ищут символы ascii в файлах, поэтому они не подходят для чтения двоичных файлов. fread читает двоичные файлы, но вы не можете сохранить его в массиве char, вместо этого мы можем go для uint8_t. После внесения этих изменений теперь он работает нормально.

Итак, вывод: все функции читаются до EOF, но тип данных, который мы использовали для их хранения, не может содержать все значения. Вот почему у меня были разные значения, когда я использовал strlen ().

Это результат, который я получил

Enter file : 1.docx
File size is : 12920 bytes (according to SEEK_END)
File size is : 12920 bytes (according to freed)
PK

, когда я изменил код на этот

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

void main(){
    printf("Enter file : ");
    char filename[50];
    scanf("%s",&filename);

    FILE *f = fopen(filename, "rb");
    fseek(f, 0, SEEK_END);
    long fsize = ftell(f);
    fseek(f, 0, SEEK_SET);  /* same as rewind(f); */

    printf("File size is : %ld bytes (according to SEEK_END)\n", fsize);

    char *string = malloc(fsize + 1);
    long len = fread(string, 1, fsize, f);
    // long len = strlen(string);

    printf("File size is : %ld bytes (according to freed)\n", len);

    fclose(f);

    string[fsize] = 0;
    puts(string);
}

что теперь имеет больше смысла.

...