인프런 영문 브랜드 로고
인프런 영문 브랜드 로고

인프런 커뮤니티 질문&답변

hahahl님의 프로필 이미지

작성한 질문수

홍정모의 따라하며 배우는 C언어

13.2 텍스트 파일 입출력 예제

인자부족으로 인한 종료

해결된 질문

작성

·

54

1

#define CRTSECURE_NO_WARNINGS

#include <stdio.h>

#include <stdlib.h>

int main(int argc, char* argv[])

{

int ch;

FILE* fr; // TODO: file pointer to write

FILE* fw;

const char* out_filename = "output.txt";

unsigned long count = 0;

if (argc != 3)

{

printf("Usage: %s filename\n", argv[0]);

exit(EXIT_FAILURE);

}

if ((fr = fopen(argv[1], "r")) == NULL) // Open a text file for reading.

{

printf("Can't open %s\n", argv[1]);

exit(EXIT_FAILURE);

}

if ((fw = fopen(argv[2], "w")) == NULL)

{

printf("Can't open %s\n", argv[2]);

exit(EXIT_FAILURE);

}

/*

r: reading

w: creating and writing or over writing

a: appending or creating and writing

r+: both reading and writing

w+: reading and writing, over writingor creaitng

a+: reading and writing, appending or creating

*/

while ((ch = fgetc(fr)) != EOF) // getc(fr)

{

fputc(ch, stdout);

fputc(ch, fw);

count++;

}

printf("\n");

fclose(fr);

fclose(fw);

printf("File %s has %lu characters\n", argv[1], count);

printf("Copied to %s", out_filename);

return 0;

}

질문) 출력 조건문에서 out_filename대신 argv[2]를 대신 사용하려면 printf("Usage: %s filename\n", argv[0]); 이 부분 조건문 부분의 인자를 3으로 해야 안전하게 데이터를 보호할 수 있는 것으로 알고 있는데 이렇게되면 파일이 없는 경우 생성이 되기전에 프로그램이 종료가 되는 문제점이 발생합니다. 이 상황에서 argv[2]를 사용하려고 하면 코드를 어떻게 구성해야 프로그램이 정상작동될까요?

답변 1

2

Soobak님의 프로필 이미지

안녕하세요? 질문&답변 도우미 Soobak 입니다.

 

좋은 고민이시네요.
우선, 말씀해주신 내용 중 '파일이 없는 경우 생성되기 전에 프로그램이 종료가 되는 문제점이 발생' 부분에서 '파일이 없는 경우' 라는 표현보다는, 'argv[2] 인수가 없는 경우', 즉, '출력하려는 파일의 이름'이 인수로 전달되지 않은 경우 로 생각해보시면, 해결 방법을 떠올리시는 것이 보다 편하실 것 같습니다.

 

문제는 다양한 방법으로 해결할 수 있을 것 같습니다.
argv[2] 를 활용하여 출력 파일의 이름을 반드시 지정하도록 변경하시는 것이 의도이시라면,
말씀해주신 내용처럼 argc3 이 아닌 경우에 대해서 Usage: 프로그램이름(argv[0]) filename 을 출력하고 프로그램을 종료하는 부분을 Usage: 프로그램이름(argv[0]) 읽을파일이름 출력할파일이름 등과 같이 수정하여, 사용자가 argv[2] 도 함께 입력해야 됨을 알 수 있도록 사용자 친화적으로 변경하는 방법이 있을 것 같습니다.

또는 일종의 '기본값' 설정처럼, argc2 인 경우, 즉, 읽어들일 파일의 이름만 인수로 전달된 경우에는 현재와 같이 "output.txt" 문자열을 생성 파일의 이름으로 사용하되, argc3 인 경우, 즉, 출력 파일의 이름 또한 인수로 전달된 경우에는 "output.txt" 대신 argv[2] 를 생성 파일의 이름으로 사용하도록 구성할 수도 있을 것 같습니다.

hahahl님의 프로필 이미지
hahahl
질문자

정확히 이해한 것인지 모르겠지만 수정해보았습니다.

#define CRTSECURE_NO_WARNINGS

#include <stdio.h>

#include <stdlib.h>

int main(int argc, char* argv[])

{

int ch;

FILE* fr; // TODO: file pointer to write

FILE* fw;

const char* out_filename = "output.txt";

unsigned long count = 0;

if (argc < 2 || argc > 3)

{

printf("Usage: %s 읽을파일이름 [출력할파일이름]\n", argv[0]);

exit(EXIT_FAILURE);

}

if ((fr = fopen(argv[1], "r")) == NULL) // Open a text file for reading.

{

printf("Can't open %s\n", argv[1]);

exit(EXIT_FAILURE);

}

if (argc == 3)

{

out_filename = argv[2];

}

if ((fw = fopen(out_filename, "w")) == NULL)

{

printf("Can't open %s\n", out_filename);

exit(EXIT_FAILURE);

}

/*

r: reading

w: creating and writing or over writing

a: appending or creating and writing

r+: both reading and writing

w+: reading and writing, over writingor creaitng

a+: reading and writing, appending or creating

*/

while ((ch = fgetc(fr)) != EOF) // getc(fr)

{

fputc(ch, stdout);

fputc(ch, fw);

count++;

}

printf("\n");

fclose(fr);

fclose(fw);

printf("File %s has %lu characters\n", argv[1], count);

printf("Copied to %s", out_filename);

return 0;

}

Soobak님의 프로필 이미지

네, 잘 이해하시고 작성하셨습니다.

강의의 내용에서 더 나아가 능동적으로 재밌게 학습하시는 것 같아, 저 또한 재미를 느끼네요. 감사합니다.

우선, 첨부해주신 코드를 컴파일 후 실행해보며 여러가지 경우를 체크해보았습니다.
이를 바탕으로, 문제가 발생할 '수' 도 있는 상황에 대해서 말씀드려봅니다.

 

const char* out_filename 에서의 자료형에 관한 문제입니다.
이렇게 const 로 한정한 포인터 변수에 대해서 추후 argv[2] 를 할당하는 것은 C언어에서 '문법적으로' 문제가 되지는 않지만, const 로 한정하는 것의 의미에 일치하지 않으며 추후 혼동을 일으킬 수 있으므로 권장되지 않는 방식입니다. (이 부분은 강의 10.5 포인터의 호환성17:50 부분의 교수님 설명을 참고해보시면 이해에 도움이 많이 되실 것 같습니다.)

따라서, chsr* out_filename 을 선언하고, 문자열을 동적으로 할당해주는 것이 보다 더 좋은 코드라고 생각합니다.

 

이미 훌륭하게 작성하셨기에, 이해를 조금 도와드리기 위한 예시 코드를 첨부드려봅니다.

이해를 위해 답변드렸던 이전 답변의 내용의 일관성을 위하여, 로그 부분은 영어가 아닌 한글로 통일하여 작성하였습니다.

 

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#define BUFFER_SIZE 4096
#define DEFAULT_OUT_FILENAME "output.txt"

int main(int argc, char* argv[])
{
    FILE* fr;
    FILE* fw;
    char buffer[BUFFER_SIZE];
    size_t bytes_read;
    char* out_filename;
    unsigned long count = 0;

    if (argc < 2 || argc > 3)
    {
        fprintf(stderr, "사용법: %s 읽을파일이름 [출력할파일이름]\n", argv[0]);
        exit(EXIT_FAILURE);
    }

    if ((fr = fopen(argv[1], "r")) == NULL)
    {
        fprintf(stderr, "%s 파일을 열 수 없습니다\n", argv[1]);
        exit(EXIT_FAILURE);
    }

    if (argc == 3)
    {
        out_filename = strdup(argv[2]);
    }
    else
    {
        out_filename = strdup(DEFAULT_OUTPUT);
    }

    if (out_filename == NULL)
    {
        fprintf(stderr, "문자열 메모리 할당 실패\n");
        fclose(fr);
        exit(EXIT_FAILURE);
    }

    if ((fw = fopen(out_filename, "w")) == NULL)
    {
        fprintf(stderr, "%s 파일을 열 수 없습니다\n", out_filename);
        free(out_filename);
        fclose(fr);
        exit(EXIT_FAILURE);
    }

    while ((bytes_read = fread(buffer, 1, BUFFER_SIZE, fr)) > 0)
    {
        fwrite(buffer, 1, bytes_read, stdout);
        fwrite(buffer, 1, bytes_read, fw);
        count += bytes_read;
    }

    printf("\n");
    fclose(fr);
    fclose(fw);

    printf("%s 파일은 %lu 바이트입니다\n", argv[1], count);
    printf("%s 파일로 복사되었습니다\n", out_filename);

    free(out_filename);
    return 0;
}

 

hahahl님의 프로필 이미지

작성한 질문수

질문하기