[c] C에서 파일의 크기를 어떻게 결정합니까?

바이트 단위로 파일 크기를 어떻게 알 수 있습니까?

#include <stdio.h>

unsigned int fsize(char* file){
  //what goes here?
}



답변

NilObject의 코드를 기반으로 :

#include <sys/stat.h>
#include <sys/types.h>

off_t fsize(const char *filename) {
    struct stat st;

    if (stat(filename, &st) == 0)
        return st.st_size;

    return -1;
}

변경 사항 :

  • 파일 이름 인수를 a로 만들었습니다 const char.
  • struct stat변수 이름이 누락 된 정의를 수정했습니다 .
  • 빈 파일에 대해서는 모호한 -1대신에 오류가 발생하면 반환 0합니다. off_t부호있는 유형이므로 가능합니다.

당신이 원하는 경우 fsize()에러 메시지를 인쇄하려면, 당신은이를 사용할 수 있습니다 :

#include <sys/stat.h>
#include <sys/types.h>
#include <string.h>
#include <stdio.h>
#include <errno.h>

off_t fsize(const char *filename) {
    struct stat st;

    if (stat(filename, &st) == 0)
        return st.st_size;

    fprintf(stderr, "Cannot determine size of %s: %s\n",
            filename, strerror(errno));

    return -1;
}

32 비트 시스템에서는이 옵션을 사용하여이를 컴파일해야하며 -D_FILE_OFFSET_BITS=64, 그렇지 않으면 off_t최대 2GB의 값만 보유합니다. 자세한 내용은 Linux에서 큰 파일 지원 의 “LFS 사용”섹션 을 참조하십시오.


답변

를 사용하지 마십시오 int. 크기가 2 기가 바이트를 초과하는 파일은 요즘 먼지처럼 일반적입니다.

를 사용하지 마십시오 unsigned int. 크기가 4 기가 바이트를 넘는 파일은 약간 덜 일반적인 먼지로 일반적입니다.

IIRC 표준 라이브러리는 off_t부호없는 64 비트 정수로 정의 되며, 이는 모든 사람이 사용해야합니다. 우리는 16 엑사 바이트 파일을 가지고 시작했을 때 몇 년 안에 128 비트로 재정의 할 수 있습니다.

Windows를 사용하는 경우 GetFileSizeEx 를 사용해야 합니다. 실제로 부호있는 64 비트 정수를 사용하므로 8 엑사 바이트 파일에 문제가 발생하기 시작합니다. 어리석은 Microsoft! 🙂


답변

Matt의 솔루션은 C 대신 C ++이라는 점을 제외하고는 작동해야하며 초기 텔은 필요하지 않습니다.

unsigned long fsize(char* file)
{
    FILE * f = fopen(file, "r");
    fseek(f, 0, SEEK_END);
    unsigned long len = (unsigned long)ftell(f);
    fclose(f);
    return len;
}

중괄호도 당신을 위해 고쳤습니다. 😉

업데이트 : 이것은 실제로 최고의 솔루션은 아닙니다. 그것은 Windows에서 기가 바이트 파일 제한 그리고 그것은 단지 같은 플랫폼 별 호출을 사용하는 것보다 가능성이 느리다 GetFileSizeEx거나 stat64.


답변

**이 작업을 수행하지 마십시오 ( 왜? ) :

온라인 내가 발견하는 C99 표준 문서를 인용 : “파일의 끝을,와 마찬가지로 파일 위치 지정자를 설정 fseek(file, 0, SEEK_END)(때문에 가능 뒤에 널 문자) 바이너리 스트림에 대한 정의되지 않은 동작이 있거나 상태에 의존하는 인코딩 어떤 스트림 초기 변속 상태에서 종료되지 않습니다. **

오류 메시지가 전송 될 수 있도록 INT, 다음 사용하도록 정의를 변경 fseek()하고 ftell()파일 크기를 결정합니다.

int fsize(char* file) {
  int size;
  FILE* fh;

  fh = fopen(file, "rb"); //binary mode
  if(fh != NULL){
    if( fseek(fh, 0, SEEK_END) ){
      fclose(fh);
      return -1;
    }

    size = ftell(fh);
    fclose(fh);
    return size;
  }

  return -1; //error
}


답변

POSIX

POSIX 표준은 파일 크기를 얻기 위해 자신의 방법이있다.
기능을 사용
하려면 sys/stat.h헤더를 포함하십시오 .

개요

  • 를 사용하여 파일 통계를 가져옵니다 stat(3).
  • st_size재산을 얻으십시오 .

참고 : 크기는로 제한됩니다 4GB. Fat32파일 시스템 이 아닌 경우 64 비트 버전을 사용하십시오!

#include <stdio.h>
#include <sys/stat.h>

int main(int argc, char** argv)
{
    struct stat info;
    stat(argv[1], &info);

    // 'st' is an acronym of 'stat'
    printf("%s: size=%ld\n", argv[1], info.st_size);
}
#include <stdio.h>
#include <sys/stat.h>

int main(int argc, char** argv)
{
    struct stat64 info;
    stat64(argv[1], &info);

    // 'st' is an acronym of 'stat'
    printf("%s: size=%ld\n", argv[1], info.st_size);
}

ANSI C (표준)

ANSI C는 직접 파일의 길이를 결정하는 방법을 제공하지 않습니다.

우리는 마음을 사용해야합니다. 지금, 우리는 탐색 접근법을 사용할 것입니다!

개요

  • 을 사용하여 파일을 끝까지 찾으십시오 fseek(3).
  • 를 사용하여 현재 위치를 가져옵니다 ftell(3).

#include <stdio.h>

int main(int argc, char** argv)
{
    FILE* fp = fopen(argv[1]);
    int f_size;

    fseek(fp, 0, SEEK_END);
    f_size = ftell(fp);
    rewind(fp); // to back to start again

    printf("%s: size=%ld", (unsigned long)f_size);
}

파일이 stdin파이프 인 경우 POSIX, ANSI C 가 작동하지 않습니다.
파일이 파이프 또는이면
반환 0됩니다 stdin.

의견 : 대신 POSIX 표준을 사용해야 합니다. 64 비트를 지원하기 때문입니다.


답변

Windows 응용 프로그램을 빌드하는 경우 특히 다른 시스템의 파일 표현의 특성으로 인해 파일 길이를 결정하기 위해 CRT 파일 I / O가 지저분 하므로 GetFileSizeEx API를 사용하십시오 .)


답변

std c 라이브러리를 사용하는 것이 좋다면 :

#include <sys/stat.h>
off_t fsize(char *file) {
    struct stat filestat;
    if (stat(file, &filestat) == 0) {
        return filestat.st_size;
    }
    return 0;
}