C언어로 파일 포인터를 활용한 파일 복사 프로그램 작성법

파일 포인터는 C언어에서 파일 처리를 위해 사용하는 핵심 개념 중 하나입니다. 이를 활용하면 파일의 내용을 효율적으로 읽거나 쓸 수 있으며, 복사와 같은 작업을 구현할 수 있습니다. 본 기사에서는 파일 포인터를 사용해 텍스트 파일과 바이너리 파일을 복사하는 프로그램을 단계적으로 작성하는 방법을 소개합니다. 파일 입출력 함수의 사용법부터 오류 처리와 프로그램 확장 가능성까지 다룰 예정이므로, 초보 프로그래머도 쉽게 따라 할 수 있습니다.

목차

파일 포인터와 파일 I/O의 기본 개념


파일 포인터는 C언어에서 파일을 조작하기 위해 사용하는 데이터 구조입니다. 이를 통해 파일의 특정 위치를 가리키며 데이터를 읽거나 쓸 수 있습니다.

파일 포인터란?


파일 포인터는 파일과 프로그램 간의 연결을 담당하는 포인터로, 표준 라이브러리에서 정의된 FILE 구조체의 주소를 저장합니다. 이를 통해 프로그램은 파일의 내용을 읽거나 쓰는 위치를 제어할 수 있습니다.

파일 I/O의 원리


파일 I/O(Input/Output)은 파일로부터 데이터를 읽고 쓰는 작업을 의미합니다. 파일 포인터를 사용하면 다음과 같은 과정을 거칩니다.

  1. 파일 열기: 파일을 열어 파일 포인터를 초기화합니다.
  2. 파일 읽기/쓰기: 파일의 데이터를 읽거나 새로운 데이터를 씁니다.
  3. 파일 닫기: 작업이 완료된 후 파일 포인터를 닫아 리소스를 해제합니다.

기본적인 파일 조작 흐름

  • fopen(): 파일을 열고 파일 포인터를 반환합니다.
  • fclose(): 열려 있는 파일을 닫습니다.
  • fread(), fwrite(): 데이터를 파일로 읽거나 씁니다.

예를 들어, 파일에서 데이터를 읽으려면 다음과 같은 코드를 사용할 수 있습니다.

#include <stdio.h>

int main() {
    FILE *file = fopen("example.txt", "r");
    if (file == NULL) {
        printf("파일 열기에 실패했습니다.\n");
        return 1;
    }

    char buffer[100];
    while (fgets(buffer, sizeof(buffer), file) != NULL) {
        printf("%s", buffer);
    }

    fclose(file);
    return 0;
}

위 코드는 파일 포인터를 통해 파일을 읽고 내용을 출력하는 기본적인 파일 I/O의 예제입니다.

fopen(), fclose() 함수의 사용법

파일 입출력을 수행하기 위해 가장 먼저 필요한 작업은 파일을 열고 닫는 것입니다. 이를 위해 C언어에서는 fopen()fclose() 함수를 제공합니다.

fopen() 함수


fopen() 함수는 파일을 열고 해당 파일에 대한 파일 포인터를 반환합니다. 사용법은 다음과 같습니다.

FILE *fopen(const char *filename, const char *mode);
  • filename: 열고자 하는 파일의 경로 및 이름입니다.
  • mode: 파일 열기 모드로, 파일의 읽기, 쓰기, 추가 등의 동작을 지정합니다.

열기 모드의 주요 옵션은 다음과 같습니다.

  • "r": 읽기 전용으로 파일 열기. 파일이 존재하지 않으면 오류 발생.
  • "w": 쓰기 전용으로 파일 열기. 파일이 존재하지 않으면 새로 생성, 존재하면 기존 내용을 덮어씀.
  • "a": 추가 모드로 파일 열기. 파일이 존재하지 않으면 새로 생성.
  • "rb", "wb", "ab": 각각 바이너리 모드에서 읽기, 쓰기, 추가 모드로 열기.

예제 코드

#include <stdio.h>

int main() {
    FILE *file = fopen("example.txt", "r");
    if (file == NULL) {
        printf("파일 열기에 실패했습니다.\n");
        return 1;
    }
    printf("파일 열기 성공!\n");
    fclose(file);
    return 0;
}

위 코드는 파일을 읽기 전용으로 열고 성공 여부를 출력한 후 파일을 닫습니다.

fclose() 함수


fclose() 함수는 열려 있는 파일 포인터를 닫고 관련 리소스를 해제합니다. 사용법은 다음과 같습니다.

int fclose(FILE *stream);
  • stream: 닫으려는 파일 포인터입니다.

예제 코드

FILE *file = fopen("example.txt", "r");
if (file != NULL) {
    // 파일 작업 수행
    fclose(file);
} else {
    printf("파일 열기에 실패했습니다.\n");
}

주의 사항

  1. 파일이 열리지 않는 경우: fopen()이 NULL을 반환하는지 반드시 확인해야 합니다.
  2. 열린 파일 닫기: 사용이 끝난 파일은 항상 fclose()로 닫아야 리소스 누수를 방지할 수 있습니다.
  3. 모드 선택: 파일의 용도에 맞는 열기 모드를 선택해야 합니다. 잘못된 모드를 사용하면 예상치 못한 결과가 발생할 수 있습니다.

위 함수는 파일 복사 프로그램을 포함한 다양한 파일 입출력 작업의 기본이 되므로 정확히 이해하고 사용해야 합니다.

fread(), fwrite()를 사용한 데이터 읽기와 쓰기

fread()fwrite() 함수는 C언어에서 파일 입출력을 수행할 때 주로 사용하는 함수로, 바이너리 데이터를 읽고 쓸 때 유용합니다. 이를 통해 텍스트뿐 아니라 이미지나 영상 같은 파일도 처리할 수 있습니다.

fread() 함수


fread() 함수는 파일에서 데이터를 읽어 지정된 버퍼에 저장합니다. 사용법은 다음과 같습니다.

size_t fread(void *ptr, size_t size, size_t count, FILE *stream);
  • ptr: 읽은 데이터를 저장할 버퍼의 포인터입니다.
  • size: 읽어들일 각 데이터 요소의 크기(바이트 단위)입니다.
  • count: 읽을 데이터 요소의 개수입니다.
  • stream: 읽을 파일을 나타내는 파일 포인터입니다.

예제 코드

#include <stdio.h>

int main() {
    FILE *file = fopen("example.bin", "rb");
    if (file == NULL) {
        printf("파일 열기에 실패했습니다.\n");
        return 1;
    }

    char buffer[256];
    size_t bytesRead = fread(buffer, sizeof(char), sizeof(buffer), file);
    printf("읽은 바이트 수: %zu\n", bytesRead);

    fclose(file);
    return 0;
}

위 코드는 파일에서 데이터를 읽어 버퍼에 저장한 후, 읽은 바이트 수를 출력합니다.

fwrite() 함수


fwrite() 함수는 버퍼에 있는 데이터를 파일에 씁니다. 사용법은 다음과 같습니다.

size_t fwrite(const void *ptr, size_t size, size_t count, FILE *stream);
  • ptr: 파일에 쓸 데이터를 저장한 버퍼의 포인터입니다.
  • size: 쓸 데이터 요소의 크기(바이트 단위)입니다.
  • count: 쓸 데이터 요소의 개수입니다.
  • stream: 데이터를 쓸 파일을 나타내는 파일 포인터입니다.

예제 코드

#include <stdio.h>

int main() {
    FILE *file = fopen("output.bin", "wb");
    if (file == NULL) {
        printf("파일 열기에 실패했습니다.\n");
        return 1;
    }

    char data[] = "Hello, World!";
    size_t bytesWritten = fwrite(data, sizeof(char), sizeof(data), file);
    printf("쓴 바이트 수: %zu\n", bytesWritten);

    fclose(file);
    return 0;
}

위 코드는 버퍼에 저장된 데이터를 파일에 쓰고, 작성한 바이트 수를 출력합니다.

fread()와 fwrite()의 차이점

  • fread()는 파일에서 데이터를 읽어 버퍼에 저장합니다.
  • fwrite()는 버퍼에 저장된 데이터를 파일에 기록합니다.

주의 사항

  1. 읽기 및 쓰기 실패 확인: 반환값을 통해 읽거나 쓴 데이터의 실제 크기를 확인해야 합니다.
  2. 버퍼 크기 관리: 파일의 크기에 따라 적절한 버퍼 크기를 설정해야 메모리 문제를 방지할 수 있습니다.
  3. 바이너리 파일 처리: 텍스트 파일과 달리 바이너리 파일은 fread()fwrite()를 사용해야 정확히 처리됩니다.

이 함수들은 파일 복사 프로그램에서 파일 데이터를 읽고 쓰는 핵심적인 역할을 수행합니다.

파일 복사 프로그램의 구조 설계

파일 복사 프로그램은 소스 파일에서 데이터를 읽어 대상 파일에 기록하는 단순한 동작을 수행합니다. 하지만 이를 효과적으로 구현하기 위해 프로그램의 구조를 체계적으로 설계해야 합니다.

프로그램 동작 흐름

  1. 파일 열기: 소스 파일을 읽기 모드로, 대상 파일을 쓰기 모드로 엽니다.
  2. 데이터 읽기 및 쓰기: 소스 파일에서 데이터를 읽어 대상 파일에 기록합니다.
  3. 파일 닫기: 소스와 대상 파일 모두 닫아 리소스를 해제합니다.
  4. 오류 처리: 파일 열기 실패나 읽기/쓰기 오류를 처리합니다.

프로그램의 주요 구성 요소

  1. 입출력 파일 경로 입력
  • 사용자로부터 소스 파일과 대상 파일의 경로를 입력받습니다.
  • 파일이 유효한지 확인합니다.
  1. 파일 처리
  • fopen()을 사용해 파일을 열고, 실패 시 오류 메시지를 출력합니다.
  • fread()fwrite()를 통해 데이터를 읽고 씁니다.
  1. 반복 작업
  • 버퍼를 사용해 데이터를 일정 크기씩 처리하여 메모리 사용을 최적화합니다.
  1. 종료 처리
  • 작업 완료 후 fclose()로 파일을 닫고 리소스를 해제합니다.

프로그램 설계 예시

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

#define BUFFER_SIZE 1024

int main() {
    char sourcePath[256], destPath[256];
    FILE *sourceFile, *destFile;
    char buffer[BUFFER_SIZE];
    size_t bytesRead;

    // 파일 경로 입력
    printf("복사할 소스 파일 경로: ");
    scanf("%s", sourcePath);
    printf("대상 파일 경로: ");
    scanf("%s", destPath);

    // 파일 열기
    sourceFile = fopen(sourcePath, "rb");
    if (sourceFile == NULL) {
        perror("소스 파일 열기 실패");
        return 1;
    }

    destFile = fopen(destPath, "wb");
    if (destFile == NULL) {
        perror("대상 파일 열기 실패");
        fclose(sourceFile);
        return 1;
    }

    // 데이터 읽기 및 쓰기
    while ((bytesRead = fread(buffer, 1, BUFFER_SIZE, sourceFile)) > 0) {
        fwrite(buffer, 1, bytesRead, destFile);
    }

    // 파일 닫기
    fclose(sourceFile);
    fclose(destFile);

    printf("파일 복사가 완료되었습니다.\n");
    return 0;
}

설계의 주요 포인트

  1. 버퍼 사용: 데이터를 일정 크기로 처리해 메모리 효율성을 높입니다.
  2. 오류 처리: 파일 열기 실패, 읽기 및 쓰기 오류를 처리합니다.
  3. 확장 가능성: 바이너리 파일 복사나 대량 데이터 처리에 쉽게 확장할 수 있습니다.

이 구조를 기반으로 파일 복사 프로그램의 코드를 작성하면 효율적이고 안정적으로 동작하는 프로그램을 구현할 수 있습니다.

파일 복사 코드 작성과 설명

파일 복사 프로그램은 소스 파일의 데이터를 읽어 대상 파일에 쓰는 과정을 반복하여 데이터를 복사합니다. 아래는 이를 구현한 C언어 코드와 각 부분에 대한 설명입니다.

전체 코드

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

#define BUFFER_SIZE 1024  // 버퍼 크기 정의

int main() {
    char sourcePath[256], destPath[256];
    FILE *sourceFile, *destFile;
    char buffer[BUFFER_SIZE];
    size_t bytesRead;

    // 사용자로부터 소스와 대상 파일 경로 입력받기
    printf("복사할 소스 파일 경로를 입력하세요: ");
    scanf("%255s", sourcePath);
    printf("대상 파일 경로를 입력하세요: ");
    scanf("%255s", destPath);

    // 소스 파일 열기
    sourceFile = fopen(sourcePath, "rb");
    if (sourceFile == NULL) {
        perror("소스 파일 열기 실패");
        return 1;
    }

    // 대상 파일 열기
    destFile = fopen(destPath, "wb");
    if (destFile == NULL) {
        perror("대상 파일 열기 실패");
        fclose(sourceFile);
        return 1;
    }

    // 소스 파일에서 데이터 읽기 및 대상 파일에 쓰기
    while ((bytesRead = fread(buffer, 1, BUFFER_SIZE, sourceFile)) > 0) {
        size_t bytesWritten = fwrite(buffer, 1, bytesRead, destFile);
        if (bytesWritten != bytesRead) {
            perror("파일 쓰기 오류");
            fclose(sourceFile);
            fclose(destFile);
            return 1;
        }
    }

    // 파일 닫기
    fclose(sourceFile);
    fclose(destFile);

    printf("파일 복사가 완료되었습니다.\n");
    return 0;
}

코드 설명

  1. 파일 경로 입력
  • scanf()를 사용하여 사용자로부터 소스 파일과 대상 파일의 경로를 입력받습니다.
  • "%255s"를 사용해 버퍼 오버플로를 방지합니다.
  1. 파일 열기
  • fopen() 함수로 소스 파일을 읽기 모드("rb")로 열고, 대상 파일을 쓰기 모드("wb")로 엽니다.
  • 파일 열기에 실패하면 perror()를 사용해 오류 메시지를 출력합니다.
  1. 데이터 읽기 및 쓰기
  • fread()로 소스 파일에서 데이터를 읽고, fwrite()로 대상 파일에 씁니다.
  • 읽은 바이트(bytesRead)와 쓴 바이트(bytesWritten)를 비교하여 쓰기 오류를 검출합니다.
  1. 반복 처리
  • while 루프를 사용하여 파일 끝에 도달할 때까지 데이터를 처리합니다.
  • 버퍼 크기(BUFFER_SIZE)만큼 데이터를 반복적으로 읽고 쓰는 방식으로 메모리 효율을 유지합니다.
  1. 파일 닫기
  • fclose()를 호출해 소스 파일과 대상 파일을 닫고 리소스를 해제합니다.
  1. 오류 처리
  • 파일 열기 실패, 읽기 실패, 쓰기 실패 등 주요 오류를 처리하여 프로그램이 비정상적으로 종료되지 않도록 설계합니다.

실행 결과 예시

복사할 소스 파일 경로를 입력하세요: source.txt
대상 파일 경로를 입력하세요: dest.txt
파일 복사가 완료되었습니다.

이 코드는 텍스트 및 바이너리 파일 모두를 복사할 수 있으며, 오류 처리와 메모리 효율성을 고려한 설계로 안정적입니다.

오류 처리와 디버깅

파일 복사 프로그램은 파일 열기 실패, 읽기/쓰기 오류, 메모리 부족 등의 문제가 발생할 수 있습니다. 이를 방지하고 프로그램의 안정성을 높이기 위해 철저한 오류 처리와 디버깅이 필요합니다.

오류 처리 방법

  1. 파일 열기 실패
  • 파일이 존재하지 않거나, 경로가 잘못되었거나, 접근 권한이 없을 경우 발생합니다.
  • fopen()의 반환값이 NULL인지 확인하여 적절한 오류 메시지를 출력합니다.
   FILE *file = fopen("example.txt", "r");
   if (file == NULL) {
       perror("파일 열기 실패");
       return 1;
   }
  1. 읽기 오류
  • 디스크 오류나 파일의 잘못된 형식 등으로 인해 발생할 수 있습니다.
  • fread()의 반환값을 확인하여 실제로 읽힌 데이터 크기를 검증합니다.
   size_t bytesRead = fread(buffer, 1, BUFFER_SIZE, file);
   if (bytesRead < BUFFER_SIZE && ferror(file)) {
       perror("파일 읽기 오류");
       fclose(file);
       return 1;
   }
  1. 쓰기 오류
  • 대상 파일이 쓰기 불가능하거나 디스크 용량이 부족할 경우 발생합니다.
  • fwrite()의 반환값과 ferror()를 활용하여 오류 여부를 확인합니다.
   size_t bytesWritten = fwrite(buffer, 1, bytesRead, destFile);
   if (bytesWritten != bytesRead) {
       perror("파일 쓰기 오류");
       fclose(destFile);
       return 1;
   }
  1. 메모리 부족
  • 버퍼 크기가 지나치게 크거나 시스템 메모리가 부족한 경우 발생합니다.
  • 동적 메모리 할당을 사용하는 경우, 할당 실패 여부를 확인해야 합니다.
   char *buffer = malloc(BUFFER_SIZE);
   if (buffer == NULL) {
       perror("메모리 할당 실패");
       return 1;
   }

디버깅 전략

  1. 로그 출력
  • 작업의 진행 상황과 변수 값을 출력하여 오류 발생 위치를 파악합니다.
   printf("소스 파일에서 %zu 바이트 읽음\n", bytesRead);
  1. gdb와 같은 디버거 사용
  • GNU 디버거(gdb)를 사용하여 프로그램의 실행 과정을 추적합니다.
  • 브레이크포인트를 설정하고 변수 값을 확인하여 문제를 진단합니다.
  1. 테스트 데이터 사용
  • 다양한 크기와 형식의 파일로 테스트하여 경계 조건을 점검합니다.
  • 예: 빈 파일, 대용량 파일, 읽기/쓰기 권한이 없는 파일 등.
  1. 단위 테스트 작성
  • 파일 열기, 읽기, 쓰기, 닫기 등 각 기능을 독립적으로 테스트합니다.

실제 코드에 적용된 오류 처리 예

FILE *sourceFile = fopen(sourcePath, "rb");
if (sourceFile == NULL) {
    perror("소스 파일 열기 실패");
    return 1;
}

FILE *destFile = fopen(destPath, "wb");
if (destFile == NULL) {
    perror("대상 파일 열기 실패");
    fclose(sourceFile);
    return 1;
}

// 파일 작업 후 반드시 닫기
fclose(sourceFile);
fclose(destFile);

결론


오류 처리는 프로그램의 안정성과 신뢰성을 높이는 중요한 과정입니다. 파일 복사 프로그램에서 발생할 수 있는 모든 가능한 오류를 고려하고, 명확한 메시지와 복구 절차를 제공하는 것이 디버깅과 유지보수의 핵심입니다.

확장 가능성: 바이너리 파일 복사와 대량 파일 처리

파일 복사 프로그램은 단순히 텍스트 파일뿐만 아니라 이미지, 동영상 등 바이너리 파일 복사와 대량 파일 처리를 지원하도록 확장할 수 있습니다. 이를 통해 실제 응용 프로그램으로 발전시킬 수 있습니다.

바이너리 파일 복사


텍스트 파일과 달리, 바이너리 파일은 모든 바이트 데이터를 그대로 복사해야 합니다. fread()fwrite()는 이 작업에 적합하며, 파일 열기 시 바이너리 모드("rb""wb")를 사용해야 합니다.

바이너리 파일 복사 코드 예시

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

#define BUFFER_SIZE 4096

int main() {
    char sourcePath[256], destPath[256];
    FILE *sourceFile, *destFile;
    char buffer[BUFFER_SIZE];
    size_t bytesRead;

    printf("복사할 바이너리 파일 경로를 입력하세요: ");
    scanf("%255s", sourcePath);
    printf("대상 파일 경로를 입력하세요: ");
    scanf("%255s", destPath);

    sourceFile = fopen(sourcePath, "rb");
    if (sourceFile == NULL) {
        perror("소스 파일 열기 실패");
        return 1;
    }

    destFile = fopen(destPath, "wb");
    if (destFile == NULL) {
        perror("대상 파일 열기 실패");
        fclose(sourceFile);
        return 1;
    }

    while ((bytesRead = fread(buffer, 1, BUFFER_SIZE, sourceFile)) > 0) {
        if (fwrite(buffer, 1, bytesRead, destFile) != bytesRead) {
            perror("파일 쓰기 오류");
            fclose(sourceFile);
            fclose(destFile);
            return 1;
        }
    }

    fclose(sourceFile);
    fclose(destFile);

    printf("바이너리 파일 복사가 완료되었습니다.\n");
    return 0;
}

주요 포인트

  1. 바이너리 모드를 사용해 정확한 데이터 복사 보장.
  2. 대용량 파일 복사를 고려해 적절한 버퍼 크기 설정.

대량 파일 처리


여러 파일을 복사하려면 파일 목록을 반복적으로 처리하는 프로그램이 필요합니다.

대량 파일 복사 코드 예시

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

#define BUFFER_SIZE 4096

void copyFile(const char *sourcePath, const char *destPath) {
    FILE *sourceFile = fopen(sourcePath, "rb");
    if (sourceFile == NULL) {
        perror("소스 파일 열기 실패");
        return;
    }

    FILE *destFile = fopen(destPath, "wb");
    if (destFile == NULL) {
        perror("대상 파일 열기 실패");
        fclose(sourceFile);
        return;
    }

    char buffer[BUFFER_SIZE];
    size_t bytesRead;
    while ((bytesRead = fread(buffer, 1, BUFFER_SIZE, sourceFile)) > 0) {
        if (fwrite(buffer, 1, bytesRead, destFile) != bytesRead) {
            perror("파일 쓰기 오류");
            break;
        }
    }

    fclose(sourceFile);
    fclose(destFile);
    printf("파일 %s에서 %s로 복사가 완료되었습니다.\n", sourcePath, destPath);
}

int main() {
    int n;
    printf("복사할 파일의 개수를 입력하세요: ");
    scanf("%d", &n);

    for (int i = 0; i < n; i++) {
        char sourcePath[256], destPath[256];
        printf("소스 파일 경로 #%d: ", i + 1);
        scanf("%255s", sourcePath);
        printf("대상 파일 경로 #%d: ", i + 1);
        scanf("%255s", destPath);

        copyFile(sourcePath, destPath);
    }

    return 0;
}

주요 포인트

  1. 파일 복사 로직을 함수로 분리하여 코드 재사용성 향상.
  2. 사용자로부터 여러 파일 경로를 입력받아 반복 처리.
  3. 복사 실패 시 적절한 메시지를 출력하여 오류 발생 위치를 명확히 함.

프로그램 확장의 장점

  1. 유연성: 텍스트 및 바이너리 파일을 모두 처리 가능.
  2. 효율성: 반복 작업을 자동화하여 생산성 향상.
  3. 실전 활용성: 대량 데이터 마이그레이션이나 백업 작업에 적합.

이와 같은 확장은 프로그램을 더욱 실용적이고 범용적으로 만들어줍니다.

실전 활용 예제와 연습 문제

파일 복사 프로그램의 개념을 확장하고 실전에서 활용할 수 있는 다양한 시나리오와 연습 문제를 제공하여 독자의 이해를 돕습니다.

실전 활용 예제

  1. 텍스트 파일 변환 및 복사
  • UTF-8 텍스트 파일을 읽어 각 줄의 앞에 번호를 추가한 후, 새로운 파일로 저장합니다.
  • 이 과정에서 fgets()fprintf()를 활용합니다.
   #include <stdio.h>

   int main() {
       FILE *sourceFile = fopen("source.txt", "r");
       FILE *destFile = fopen("numbered.txt", "w");
       if (sourceFile == NULL || destFile == NULL) {
           perror("파일 열기 실패");
           return 1;
       }

       char line[256];
       int lineNumber = 1;
       while (fgets(line, sizeof(line), sourceFile) != NULL) {
           fprintf(destFile, "%d: %s", lineNumber++, line);
       }

       fclose(sourceFile);
       fclose(destFile);
       printf("텍스트 파일 복사 및 변환 완료.\n");
       return 0;
   }
  1. 이미지 파일 복사
  • .jpg 이미지를 복사하여 새로운 파일로 저장합니다.
  • 기존 바이너리 복사 코드를 그대로 활용할 수 있습니다.
   // 기존 바이너리 파일 복사 코드 활용
  1. 로그 파일 아카이브
  • 현재 디렉토리의 모든 .log 파일을 날짜별 아카이브 디렉토리로 복사합니다.
  • 디렉토리 생성 및 파일 복사 기능을 추가합니다.

연습 문제

  1. 문제 1: 텍스트 파일 합치기
  • 두 개의 텍스트 파일을 읽어 하나의 파일로 합치는 프로그램을 작성하세요.
  • 파일 끝에 \n을 추가하여 가독성을 유지합니다.
  • 예: file1.txt, file2.txtmerged.txt
  1. 문제 2: 특정 크기 이하의 파일 복사
  • 파일 크기를 확인하여 특정 크기(예: 1MB) 이하인 파일만 복사하는 프로그램을 작성하세요.
  • stat()를 사용해 파일 크기를 확인합니다.
  1. 문제 3: 파일 내용 암호화 및 복사
  • 소스 파일의 데이터를 단순한 암호화(예: 시저 암호)를 적용한 뒤 복사하세요.
  • fgetc()fputc()를 활용하여 바이트 단위로 처리합니다.
  1. 문제 4: 대량 파일의 복사 결과 로그 생성
  • 여러 파일을 복사한 후 성공 여부를 기록한 로그 파일을 작성하세요.
  • 복사 성공 파일과 실패 파일 목록을 별도로 구분합니다.
  1. 문제 5: 파일 비교 프로그램 작성
  • 두 파일의 내용을 비교하여 동일한지 확인하는 프로그램을 작성하세요.
  • 내용이 다를 경우 첫 번째 다른 바이트의 위치를 출력합니다.

결론


이러한 예제와 연습 문제를 통해 파일 복사 프로그램을 단순 복사 기능에서 실전 활용 가능한 다양한 도구로 확장할 수 있습니다. 각 문제를 해결하면서 파일 입출력과 메모리 관리에 대한 깊은 이해를 얻을 수 있습니다.

요약

본 기사에서는 C언어를 사용해 파일 포인터를 활용한 파일 복사 프로그램을 작성하는 방법을 설명했습니다. 파일 입출력의 기본 원리와 fopen(), fread(), fwrite() 등 주요 함수를 활용해 텍스트 및 바이너리 파일을 효율적으로 복사하는 방법을 다뤘습니다.

또한, 오류 처리와 디버깅 전략, 프로그램 확장 가능성, 실전 활용 예제, 연습 문제 등을 통해 실무에서 활용할 수 있는 지식을 제공했습니다. 이를 통해 독자는 파일 입출력의 기초부터 고급 응용까지 폭넓은 이해를 얻을 수 있을 것입니다.

목차