제9장 C 프로그래밍 환경 TOPCIT (Test of Practical Competency in IT)
9.1 컴파일러
gcc 컴파일러 gcc(GNU cc) 컴파일러 상업용 C 컴파일러(cc) 컴파일 -c 옵션 -o 옵션 $ gcc [-옵션] 파일 $ cc [-옵션] 파일 컴파일 $ gcc longest.c $ a.out // 실행 -c 옵션 $ gcc -c longest.c -o 옵션 $ gcc -o longest longest.o 혹은 $ gcc -o longest longest.c $ longest // 실행
단일 모듈 프로그램:longest.c debugging: strlen() debugging: gets() #include <stdio.h> #define MAXLINE 100 void copy(char from[], char to[]); char line[MAXLINE]; // 입력 줄 char longest[MAXLINE]; // 가장 긴 줄 /*입력 줄 가운데 가장 긴 줄 프린트 */ main() { int len; int max; max = 0; while (gets(line) != NULL) { len = strlen(line); if (len > max) { max = len; copy(line, longest); } if (max > 0) // 입력 줄이 있었다면 printf("…The logest…\n%s\n", longest); return 0; #include "copy.h" /* copy: from을 to에 복사; to가 충분히 크다고 가정*/ void copy(char from[], char to[]) { int i; i = 0; while ((to[i] = from[i]) != '\0') ++i; } debugging: strlen() debugging: gets()
다중 모듈 프로그램 단일 모듈 프로그램 다중 모듈 프로그램 코드의 재사용(reuse)이 어렵고, 여러 사람이 참여하는 프로그래밍이 어렵다 예를 들어 다른 프로그램에서 copy 함수를 재사용하기 힘들다 다중 모듈 프로그램 여러 개의 .c 파일들로 이루어진 프로그램 일반적으로 복잡하며 대단위 프로그램인 경우에 적합
다중 모듈 프로그램: 예 main 프로그램과 copy 함수를 분리하여 별도 파일로 작성 컴파일 main.c copy.c copy.h // 함수의 프로토타입을 포함하는 헤더 파일 컴파일 $ gcc -c main.c $ gcc -c copy.c $ gcc -o main main.o copy.o 혹은 $ gcc -o main main.c copy.c
main.c copy.c #include <stdio.h> #include <string.h> #include "copy.h" char line[MAXLINE]; // 입력 줄 char longest[MAXLINE]; // 가장 긴 줄 /*입력 줄 가운데 가장 긴 줄 프린트 */ main() { int len; int max; max = 0; while (fgets(line, MAXLINE, stdin) != NULL) { len = strlen(line); if (len > max) { max = len; copy(line, longest); } if (max > 0) // 입력 줄이 있었다면 printf("…The logest…\n%s\n", longest); return 0; #include "copy.h" /* copy: from을 to에 복사; to가 충분히 크다고 가정*/ void copy(char from[], char to[]) { int i; i = 0; while ((to[i] = from[i]) != ‘\0’) ++i; } copy.h #define MAXLINE 100 void copy(char from[], char to[]);
9.2 make 시스템
make 시스템의 필요성 다중 모듈 프로그램을 구성하는 일부 파일이 변경된 경우? 예 make 시스템 변경된 파일만 컴파일하고, 파일들의 의존 관계에 따라서 필요한 파일만 다시 컴파일하여 실행 파일을 만들면 좋다. 예 copy.c 소스 코드를 수정 목적 파일 copy.o 생성 실행파일을 생성 make 시스템 대규모 프로그램의 경우에는 헤더, 소스 파일, 목적 파일, 실행 파일의 모든 관계를 기억하고 체계적으로 관리하는 것이 필요 make 시스템을 이용하여 효과적으로 작업
메이크파일 메이크파일 $ make [-f 메이크파일] 실행 파일을 만들기 위해 필요한 파일들과 그들 사이의 의존 관 계, 만드는 방법 등을 기술 make 시스템은 메이크파일을 이용하여 파일의 상호 의존 관계를 파악하여 실행 파일을 쉽게 다시 만듬 $ make [-f 메이크파일] 옵션이 없으면 Makefile 혹은 makefile을 사용
메이크파일의 구성 메이크파일의 구성 형식 예: Makefile make 실행 대상리스트: 의존리스트 명령리스트 main: main.o copy.o gcc -o main main.o copy.o main.o: main.c copy.h gcc -c main.c copy.o: copy.c copy.h gcc -c copy.c make 실행 $ make 혹은 $ make main gcc -c main.c gcc -c copy.c gcc -o main main.o copy.o copy.c 파일이 변경된 후 $ touch copy.c $ make
9.3 디버거
gdb 가장 대표적인 디버거 gdb 사용을 위한 컴파일 gdb 주요 기능 gdb 실행 GNU debugger(gdb) 정지점(breakpoint) 설정 한 줄씩 실행 변수 접근 및 수정 함수 탐색 추적(tracing) gdb 사용을 위한 컴파일 -g 옵션을 이용하여 컴파일 $ gcc -g -o longest longest.c 다중 모듈 프로그램 $ gcc -g -o main main.c copy.c gdb 실행 $ gdb [실행파일] $ gdb main
gdb 기능 소스보기 : l(ist) l [줄번호] 지정된 줄을 프린트 l [파일명]:[함수명] 지정된 함수를 프린트 set listsize n 출력되는 줄의 수를 n으로 변경 (gdb) l copy 1 #include <stdio.h> 2 #include "copy.h" 3 /* copy: from을 to에 복사; to가 충분히 크다고 가정*/ 4 void copy(char from[], char to[]) 5 { 6 int i; 7 i = 0; 8 while ((to[i] = from[i]) != '\0') 9 ++i; 10 }
gdb 기능 정지점 : b(reak), clear, d(elete) b [파일:]함수 파일의 함수 시작부분에 정지점 설정 b n n번 줄에 정지점을 설정 b +n 현재 줄에서 n개 줄 이후에 정지점 설정 b -n 현재 줄에서 n개 줄 이전에 정지점 설정 info b 현재 설정된 정지점을 출력 clear 줄번호 해당 정지점을 삭제 d 모든 정지점을 삭제 (gdb) b 12 Breakpoint 1 at 0x8048426: file main.c, line 12. (gdb) b copy Breakpoint 2 at 0x80484c2: file copy.c, line 6. (gdb) info b Num Type Disp Enb Address What 1 breakpoint keep y 0x08048426 in main at main.c:12 2 breakpoint keep y 0x080484c2 in copy at copy.c:6
gdb 기능 프로그램 수행 변수 값 프린트: p(rint) r(un) 인수 명령줄 인수를 받아 프로그램 수행 k(ill) 프로그램 수행 강제 종료 n(ext) 멈춘 지점에서 다음 줄을 수행하고 멈춤 s(tep) n과 같은 기능 함수호출 시 함수내부로 진입 c(ontinue) 정지점을 만날 때 까지 계속 수행 u 반복문에서 빠져나옴 finish 현재 수행하는 함수의 끝으로 이동 return 현재 수행중인 함수를 빠져나옴 quit 종료 변수 값 프린트: p(rint) p [변수명] 해당 변수 값 프린트 p 파일명::[변수명] 특정 파일의 전역변수 프린트 p [함수명]::[변수명] 특정 함수의 정적 변수 프린트 info locals 현재 상태의 지역변수 리스트
gdb 기능 (gdb) r Starting program: /home/mysung/ulprog/main Breakpoint 1, main () at main.c:12 12 while (fgets(line, MAXLINE, stdin) != NULL) { (gdb) s _IO_fgets (buf=0x8049880 "", n=100, fp=0x49c125a0) at iofgets.c:37 37 { (gdb) n 42 if (n <= 0) (gdb) c Continuing. Merry X-mas ! Breakpoint 2, copy (from=0x8049880 "Merry X-mas !\n", to=0x8049800 "") at copy.c:6 6 i = 0; 7 while ((to[i] = from[i]) != '\0') (gdb) p from $8 = 0x8049880 "Merry X-mas !\n" (gdb) p to $9 = 0x8049800 "" 8 ++i; $10 = 0x8049800 "M"
gdb 기능 (gdb) c Continuing. Happy New Year ! Breakpoint 2, copy (from=0x8049880 "Happy New Year !\n", to=0x8049800 "Merry X-mas !\n") at copy.c:6 6 i = 0; (gdb) p from $11 = 0x8049880 “Happy New Year !\n” (gdb) n 7 While ((to[i] = from[i]) != ‘\0’) (gdb) p to $12 = 0x8049800 "Merry X-mas !\n" 8 ++i; $13 = 0x8049800 "Herry X-mas !\n“ ^D …The longest… [Inferior 1 (process 17746) exited normally]
DDD(Data Display Debugger) gdb를 위한 그래픽 사용자 인터페이스 http://www.gnu.org/software /ddd 정지점을 설정 소스코드의 원하는 위치에 커 서를 이동하고 상 Break 버튼 Next나 Step 같은 명령어 버 튼을 이용하여 한 줄씩 실행 하단에는 gdb 명령어 입력 창
Next: 한 줄 진행
Cont: 계속 진행
Up: 함수 호출자 보기