46
1 Part 14 파파 파파파

Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

  • Upload
    -

  • View
    110

  • Download
    2

Embed Size (px)

Citation preview

Page 1: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

1

Part 14 파일 입출력

Page 2: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

2

이 장의 내용 파일과 파일 포인터 파일 입출력 함수 이진 파일 임의 접근 파일 처리

Page 3: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

3

14.1 파일과 파일 포인터

Page 4: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

4

파일 C 의 파일은 모든 데이터를 연속된 바이트 형태로

저장한다 .

Page 5: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

5

C 언어의 파일 종류 텍스트 파일 (text file)

사람들이 읽을 수 있는 문자들을 저장하고 있는 파일 텍스트 파일에서 “한 줄의 끝”을 나타내는 표현은 파일이 읽어

들여질 때 , C 내부의 방식으로 변환된다 .

이진 파일 (binary file) 모든 데이터는 있는 그대로 바이트의 연속으로 저장 이진 파일을 이용하여 메모리에 저장된 변수 값 형태 그대로

파일에 저장할 수 있다 .

Page 6: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

6

파일 열기

파일을 사용하기 위해서는 반드시 먼저 파일 열기 (fopen) 를 해야 한다 . 파일 열기 (fopen) 를 하면 FILE 구조체에 대한

포인터가 리턴된다 . FILE 포인터는 열린 파일을 지정한다 .

Page 7: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

7

FILE 구조체 FILE 구조체는 stdio.h 에 정의되어 있음 . FILE 구조체에는 열려진 파일의 현재 상태를 나타내는 필드 변수들 특히 파일 입출력에 사용되는 버퍼 관련 변수들

typedef struct {     short level;         // 버퍼의 fill/empty 수준     unsinged flags;         // 파일 상태 플래그     char  fd;            // 열린 파일 번호 (descriptor)     unsigned char hold;  // ungetc char if no buffer     short  bsize;          // 버퍼 크기     unsigned char *buffer;       // 버퍼     unsigned char *curp;          // 현재 활성화된 포인터     unsinged istemp;       // 임시 파일 지시자     short  token;          // 유효성 검사에 사용 }  FILE;   // FILE 구조체

Page 8: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

8

표준 입출력 stdin, stdout, stderr

각각 표준 입력 , 표준 출력 , 표준 오류를 나타내는 FILE 포인터

C 프로그램이 실행되면 자동적으로 열리고 프로그램이 종료될 때 자동으로 닫힘 .표준 입출력

포인터설명 가리키는

장치

stdin 표준 입력에 대한 FILE 포인터 키보드

stdout 표준 출력에 대한 FILE 포인터 모니터

stderr 표준 오류에 대한 FILE 포인터 모니터

Page 9: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

9

파일 열기 함수 fopen()

FILE *fopen(const char *filename, const char *mode); const char *filename: 파일명에 대한 포인터 const char *mode: 모드로 파일을 여는 형식

예FILE *fp; fp = fopen("c:\work\text.txt", "r"); if (fp == NULL) {     printf(" 파일 열기 오류 \n"); }

예 fp = fopen("outdata.txt", "w"); fp = fopen("outdata.txt", "a");

Page 10: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

10

fopen 모드모드 내 용

"r" 읽기 전용으로 연다

"w" 쓰기 전용으로 연다 .   파일이 없으면 새로 생성하고 이미 존재하면 그 파일 내용을 삭제

"a" 추가용으로 연다 . 파일이 없으면 새로 생성한다 .

"r+" 이미 존재하는 파일을 읽기쓰기 ( 갱신 ) 용으로 연다 .

"w+" 파일을 생성하고 갱신용으로 연다 .

"a+" 파일을 추가용 , 갱신용으로 연다 . 파일이 없으면 새로 생성한다 .

"rb" 이진 파일을 읽기용으로 연다 .

"wb" 이진 파일을 쓰기용으로 연다 .

"ab" 이진파일을 추가용으로 연다 . 파일이 없으면 새로 생성한다 .

"rb+" 이미 존재하는 이진파일을 갱신용으로 연다 .

"wb+" 이진파일을 생성하고 갱신용으로 연다 .

"ab+" 이진 파일을 추가용 , 갱신용으로 연다 . 파일이 없으면 새로 생성한다 .

Page 11: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

11

파일 닫기 파일을 열어서 사용한 후에는 파일을 닫아야 한다 .

int fclose(FILE *fp ); fp 는 fopen 함수에서 받았던 포인터 닫기에 성공하면 0, 오류일 때는 EOF( -1) 를 리턴한다 .

예 fclose(fp);

Page 12: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

12

14.2 파일 입출력 함수

Page 13: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

13

파일 입출력 함수

표준입출력함

표준 파일 입출력 함수 기능

getchar() fgetc(), getc() 문자단위로 입력하는 함수

putchar() fputc(), putc() 문자단위로 출력하는 함수

gets() fgets() 문자열을 입력하는 함수

puts() fputs() 문자열을 출력하는 함수

scanf() fscanf() 자료형에 따라 자료를 입력하는 함수

printf() fprintf() 자료형에 따라 자료를 출력하는 함수

Page 14: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

14

문자 단위 입출력 fgetc() 함수와 fputc() 함수

파일에 문자 단위 입출력을 할 수 있다 .

int fgetc(FILE *fp); getc 함수는 fp 가 지정한 파일에서 한 문자를 읽어서 리턴한

다 . 파일 끝에 도달했을 경우에는 EOF(-1) 를 리턴한다 .

int fputc(int c, FILE *fp); putc 함수는 파일에 한 문자씩 출력하는 함수 리턴값으로 출력하는 문자 리턴 출력시 오류가 발생하면 EOF(-1) 리턴

Page 15: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

15

copy1.c

실행결과 :이제 거의 다 왔습니다 . 이제 거의 다 왔습니다 . 마지막까지 최선을 다해서 C 언어를 완전 정복합시다 . 마지막까지 최선을 다해서 C 언어를 완전 정복합시다 . ^Z

Page 16: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

16

copy2.c

실행결과 :이제 거의 다 왔습니다 .

마지막까지 최선을 다해서 C 언어를 완전 정복합시다 . ^Z message.txt 파일에 저장 완료 .

Page 17: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

17

fileCopy1.c

Page 18: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

18

기타 파일 관련 함수 int feof(FILE  *fp)

파일 포인터 fp 가 파일의 끝을 탐지하면 0 이 아닌 값을 리턴하고 파일 끝이면 0 을 리턴 한다 .

int ungetc(int c, FILE *p) c 에 저장된 문자를 입력 스트림에 반납한다 . 마치 문자를 읽지

않은 것처럼 파일 위치 지정자를 1 감소시킨다 .

int fflush(FILE *fp) 아직 기록되지 않고 버퍼에 남아 있는 데이터를 fp 가 가리키는

출력 파일에 보낸다 . 버퍼 비우기 기능을 수행하는 함수이다 .

Page 19: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

19

줄 단위 입출력 fgets() 함수와 fputs() 함수

텍스트 파일에서 한 줄씩 읽거나 쓸 수 있다 . char* fgets(char *s, int n, FILE *fp);

파일로부터 한 줄을 읽어서 문자열 포인터 s 에 저장하고 s 를 리턴 개행문자 ('\n') 나 EOF 를 만날 때까지 파일로부터 최대 n-1 개의

문자를 읽고 읽어온 데이터의 끝에는 NULL 문자를 붙여준다 . 파일을 읽는 중 파일 끝 혹은 오류가 발생하면 NULL 포인터 리턴 .

int fputs(const char *s, FILE *fp); 문자열 s 를 파일 포인터 fp 가 가리키는 파일에 출력 성공적으로 출력한 경우에는 출력한 바이트 수를 리턴 출력할 때 오류가 발생하면 EOF 값을 리턴

Page 20: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

20

deleteLine.c

Page 21: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

21

포맷 입출력 fprintf() 함수

printf() 함수와 같은 방법으로 파일에 데이터를 츨력할 수 있다 . fscanf() 함수

scanf() 함수와 같은 방법으로 파일로부터 데이터를 읽어 들일 수 있다 .

int fprintf(FILE *fp, const char *format, ...); fprintf 함수의 첫 번째 인수 fp 는 츨력할 파일에 대한 FILE 포인터 두 번째부터의 인수는 printf 함수와 동일

int fscanf(FILE *fp, const char *format, ...); fscanf 함수의 첫 번째 인수 fp 는 입력받을 파일에 대한 FILE 포인터 두 번째부터의 인수는 scanf 함수와 동일

Page 22: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

22

wordappend.c

Page 23: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

23

실행결과 :C:> wordAppend message.txt 파일에 추가할 문장을 입력하세요 . 끝내려면 줄 시작에 엔터 키를 치세요 . 이제 C 언어가 뭔지 좀 알 것 같네요 . 감사합니다 .

파일 내용 : 이제 거의 다 왔습니다 . 이제 C 언어가 뭔지 좀 알 것 같네요 . 감사합니다 .

Page 24: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

24

14.3 이진 파일

Page 25: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

25

이진 파일 이진 파일 (Binary File)

메모리에서 표현되는 그대로 저장된 파일 파일 형식에 맞추어 읽어야 판독이 가능 실행 파일 , 사운드 파일 , 이미지 파일 등 이진 파일 입출력을 위해서는 이진 입출력 모드로 열어야 함

“rb”, “wb”, “ab”, “rb+”, “wb+”, “ab+” 등

© 우균 , 창병모

Page 26: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

26

블록 단위 입출력 fread() 와 fwrite()

한번에 일정한 크기의 데이터를 파일에 읽거나 쓰기 위한 입출력 함수

int fread(void *buf, int size, int n, FILE *fp); fp 가 가리키는 파일에서 size 크기의 블록 ( 연속된 바이트 ) 을 n 개

읽어서 버퍼 포인터 buf 가 가리키는 곳에 저장 읽어온 블록의 개수를 리턴

int fwrite(const void *buf, int size, int n, FILE *fp); 파일 포인터 fp 가 지정한 파일에 버퍼 buf 에 저장되어 있는 size

크기의 블록 ( 연속된 바이트 ) 을 n 개 기록 성공적으로 출력한 블록 개수를 리턴

Page 27: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

27

fileCopy2.c

Page 28: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

28

블록 입출력 기본 아이디어

어떤 자료형의 데이터이던지 그 데이터를 연속된 바이트로 해석해서 파일에 저장

파일에 저장된 데이터를 연속된 바이트 형태로 읽어서 원래 자료형 변수에 순서대로 저장하여 원래 데이터를 그대로 복원

예 : int 저장int i; FILE *fp = fopen("intfile", "wb"); fwrite(&i, sizeof(int), 1, fp);

Page 29: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

29

student.h

Page 30: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

30

student1.c

실행결과 :학번 이름 학년 전공 200601001 아름이 1 컴퓨터과학 200601002 다람이 1 컴퓨터과학 ... ^Z

Page 31: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

31

student2.c

Page 32: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

32

실행 결과

실행결과 :----------------------------------- 학번 이름 학년 전공 -----------------------------------     0601001   아름이 1  컴퓨터과학     0601002   다람이 1  컴퓨터과학               ...

-----------------------------------

Page 33: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

33

14.4 임의 접근 파일 처리

Page 34: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

34

파일 내 위치 파일 위치 (file position)

열린 파일에서 다음 읽거나 기록할 파일 내 위치

파일 위치 지정자 (file position indicator) 시스템 내에 그 파일의 파일 위치를 저장하고 있다 .

                                                                             그림 14.3 파일 위치

Page 35: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

35

파일 위치 관련 함수 fseek(FILE *fp, long offset, int mode)

파일 위치 지정자를 임의로 설정할 수 있는 함수이다 . rewind(FILE *fp)

파일 위치를 파일 시작점에 위치시켜 처음부터 다시 읽을 수 있도록

ftell(FILE *fp) 파일의 현재 파일 위치를 나타내는 파일 위치 지정자 값 리턴

Page 36: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

36

파일 위치 관련 함수 fseek(FILE *fp, long offset, int mode)

FILE 포인터 fp 가 가리키고 파일의 파일 위치를 모드 (mode) 기준으로 오프셋 (offset) 만큼 옮긴다 .

예 fseek(fp, 0L, SEEK_SET)  파일처음으로 이동 fseek(fp, 100L, SEEK_CUR)   현재 위치에서 100 바이트 우로

이동 fseek(fp, 0L, SEEK_END)   파일 끝으로 이동

기호 값 의미

SEEK_SET 0 파일 시작

SEEK_CUR 1 현재 위치

SEEK_END 2 파일 끝

Page 37: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

37

파일 위치 관련 함수 rewind(FILE *fp)

파일 위치를 파일 시작점에 위치시켜 처음부터 다시 읽을 수 있도록 한다 .

ftell(FILE *fp) 파일의 현재 파일 위치를 나타내는 파일 위치 지정자 값 리턴

Page 38: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

38

student3.c

Page 39: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

39

student4.c

Page 40: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

40

Key Point

Page 41: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

41

Key Point 1 파일은 모든 데이터를 연속된 바이트 형태로 저장한다 . 파일을 사용하기 위해서는 반드시 파일 열기 (fopen) 를

먼저 해야 한다 . 파일 열기 (fopen) 를 하면 FILE 구조체에 대한

포인터가 리턴된다 . FILE 포인터는 열린 파일을 지정한다 . fgetc() 함수와 fputc() 함수

파일에 문자 단위 입출력을 할 수 있다 . fgets() 함수와 fputs() 함수

텍스트 파일에서 한 줄씩 읽거나 쓸 수 있다 .

Page 42: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

42

Key Point 2 fprintf() 함수

printf() 함수와 같은 방법으로 파일에 데이터를 츨력할 수 있다 . fscanf() 함수

scanf() 함수와 같은 방법으로 파일로부터 데이터를 읽어 들일 수 있다 .

fread() 와 fwrite() 한번에 일정한 크기의 데이터를 파일에 읽거나 쓰기 위한 입출력 함수

파일 위치 (file position) 열린 파일에서 다음 읽거나 기록할 파일 내 위치

파일 위치 지정자 (file position indicator) 시스템 내에 그 파일의 파일 위치를 저장하고 있다 .

fseek() 함수 파일 위치 지정자를 임의로 설정할 수 있는 함수이다 .

Page 43: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

43

프로그래밍 실습

Page 44: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

44

▶ 프로그래밍 실습 도서관리 시스템

도서에 대한 데이터 입력과 도서에 대한 열람 기능을 제공하는 간단한 도서관리 시스템을 구현해보자 .

이 시스템은 도서 입력과 도서 열람을 위한 두 개의 프로그램으로 구성된다 .

1. 헤더 파일 book.h 작성 12 장에서 정의한 struct book 을 확장해서 book.h 에 새로운

구조체를 작성한다 .

Page 45: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

45

▶ 프로그래밍 실습 2. 도서 입력 프로그램

도서관에 새로운 책이 들어올 때마다 책에 대한 정보를 입력받아 파일에 저장한다 . 책에 대한 정보는 이름 , 출판일 , 저자 , 가격 등이며 책이 들어오는 순서에 따라 일련번호를 붙인다 .

2.1 book.h 헤더 파일을 include 한다 . 2.2 FILE 포인터 , struct book 타입의 필요한 변수를 선언한

다 . 2.3 fopen() 함수를 이용하여 도서 정보를 저장하기 위한

파일을 연다 . 2.4 scanf() 함수를 이용하여 새로 들어온 책에 대한 정보를

입력 받아 구조체 변수에 저장한다 . 2.5 fwrite() 함수를 이용하여 구조체 변수에 있는 책 정보를

파일에 저장한다 . 2.6 2.3 부터 2.5 의 과정을 더 이상 입력이 없을 때까지

반복한다 .

Page 46: Part14 %ed%8 c%8c%ec%9d%bc%ec%9e%85%ec%b6%9c%eb%a0%a5

46

▶ 프로그래밍 실습 3. 도서 열람 프로그램

책의 일련번호를 입력받아 해당 책 정보를 찾아 츨력해 주는 프로그램을 작성한다 .

3.1 book.h 헤더 파일을 include 한다 . 3.2 FILE 포인터 , struct book 타입의 필요한 변수를 선언한

다 . 3.3 fopen() 함수를 이용하여 도서 정보를 저장하기 위한

파일을 연다 . 3.4 책 일련번호를 입력받고 해당 책의 레코드 위치를 계산한

다 . 3.5 fread() 함수를 이용하여 해당 레코드를 읽어서 구조체

변수에 저장한다 . 3.6 읽어 온 책에 대한 정보를 서식에 맞게 츨력한다 . 3.7 3.3 부터 3.6 까지 과정을 반복한다 .