6 장배열, 포인터, 문자열 박종혁교수 UCS Lab Tel: 970-6702 Email: jhpark1@seoultech.ac.kr SeoulTech 2012-1 st 프로그래밍입문 (1)
2 목차 6.1 1 차원배열 6.2 포인터 6.3 참조에의한호출 6.4 배열과포인터의관계 6.5 포인터연산과원소크기 6.6 함수인자로서의배열 6.7 예제 : 버블정렬 6.8 calloc() 과 malloc() 을이용한동적메모리할당 6.9 예제 : 합병과합병정렬 6.10 문자열 6.11 표준라이브러리에있는문자열조작함수 6.12 다차원배열 6.13 포인터배열 6.14 main() 함수의인자 6.15 래기드배열 6.16 인자로서의함수 6.17 예제 : 함수의근을구하기위한이분법의사용 6.18 함수포인터의배열 6.19 형한정자 const 와 volatile
3 1 차원배열 배열 : 첨자가붙은변수를사용하고여러개의동질적값을표현핛수있는자료형 예 ( 성적처리를위핚변수선얶 ) int int 1 차원배열선얶 int grade0, grade1, grade2; grade[3]; a[size]; lower bound = 0 upper bound = size - 1 size = upper bound + 1
4 1 차원배열 사용예 #define N 100 int a[n]; for (i = 0; i < N; ++i) sum += a[i];
5 1 차원배열 배열은배열명과변수의개수, 변수의자료형으로선얶한다. 배열을선얶하면변수의개수맊큼연속된기억공갂을핛당핚다.
6 1 차원배열 배열의기억공갂을사용할때는각기억공갂이배열에서차지하는위치를사용한다. 배열을구성하는기억공갂들을배열의요소 (element) 라고하며각요소 가배열에서차지하는위치를첨자 (index) 라고핚다. 배열의첨자는 0 부터시작핚다.
7 배열의초기화 배열은자동, 외부, 정적기억영역클래스는될수 있지맊, 레지스터는될수없음 젂통적인 C 에서는외부와정적배열맊배열초기 자를사용하여초기화핛수있음 ANSI C 에서는자동배열도초기화될수있음
8 배열의초기화 초기화예제 float f[5] = {0.0, 1.0, 2.0, 3.0, 4.0; f[0] = 0.0, f[1] = 1.0,... 초기자목록이초기화되는배열원소개수보다적다면, 나머지원소들은 0 으로초기화됨 int a[100] = {10 ; a[0] = 10, a[1] = 0, a[2] = 0,..., a[99] = 0 Cf) 배열초기화목록의원소개수가배열크기보다맋을경우 오류로갂주 int x[3] = {1, 3, 5, 7, 9
9 배열의초기화 외부와정적배열이명시적으로초기화되지않았다면, 시스템은디폴트로모든원소를 0으로초기화함
10 배열의초기화 배열의크기가기술되어있지않고일련의값으로초기화되도록선얶되어있다면, 초기자의개수가배열의암시적인크기가됨 int a[] = {2, 3, 5, -7; int a[4] = {2, 3, 5, -7; 따라서, 위의두선얶문은같은선얶문임
11 배열의초기화 문자열에서는주의를요함 char s[] = "abc"; 이선얶문은다음과같음 char s[] = {'a', 'b', 'c', \0'; 즉, s 배열의크기는 3이아니라 4임 문자열을저장할때끝을표시하는널문자 ( \0 ) 를 넣어줘야함.
12 배열의초기화 문자배열의마지막에널문자 ( \0 ) 가없으면저장된문자열의끝을알수없다. - printf 함수로출력하는경우쓰레기값이출력된다. char word[50]; word[0]= L ; word[1]= o ; word[2]= v ; word[3]= e ; printf( %s, word); // 넉넉핚배열선얶 // 문자를하나씩입력하여문자열완성 // 문자열출력 Love 儆儆儆儆儆儆儆儆儆儆儆儆儆儆儆儆儆
13 배열의초기화 문자배열에문자열을저장할때는항상마지막에끝을표시하는널문자 ( \0 ) 를넣어줘야한다. char word[50]; word[0]= L ; word[1]= o ; word[2]= v ; word[3]= e ; word[4]= \0 ; printf( %s, word); // 문자열의끝을널문자로표시해준다. Love
14 scanf 함수를사용한문자열의입력 문자배열에문자열을입력받을때는 %s 변환문자열과배열명을 scanf 함수의젂달인자로준다. scanf 함수로문자열을입력받으면널문자를자동으로채워준다. - word 배열에 vitamin 을입력받은경우 printf( 문자열을입력하세요 : ); scanf( %s, word); 문자열을입력하세요 : vitamin ( 엔터 )
15 scanf 함수로문자열을입력할때주의할점 배열의크기보다입력되는문자열의크기가더크면할당되지않은기억공갂을침범하므로주의해야한다. char word[5]; // 이곳에 vitamin 을입력받는다면
16 첨자 a가배열이면, a의원소를접귺핛때 a[expr] 와같이씀 여기서, expr은정수적수식이고, expr을 a의첨자, 또는색인이라고함
17 첨자 사용예 int i, a[n]; 여기서 N은기호상수이고, 하나의배열원소를참조하기위해서는첨자 i 에적당한값을할당한후 a[i] 수식을사용하면됨 이때, i 는 0 보다크거나같고 N - 1 보다작거나같아야함 i 가이범위를벖어나는값을갖는다면, a[i] 를접근할때실행시갂오류가발생함
18 포인터 프로그램에서메모리를접귺하고주소를다루기위해사용 주소연산자 & v 가변수라면, &v 는이변수의값이저장된메모리위치, 또는주소임 포인터변수 값으로주소를갖는변수 선얶방법 int *p; 즉, 변수이름앞에 * 를붙여서선얶함
19 포인터 변수를선얶하는것은메모리에기억공갂을할당하는것이며할당된 이후에는변수명으로그기억공갂을사용한다. 핛당된기억공갂을사용하는방법에는변수명외에메모리의실제 주소값을사용하는것이다. 이주소값을포인터라고핚다.
20 포인터 메모리에는바이트 (byte) 단위로그위치를식별할수있는물리적인주소값이있다. - 메모리의용량이 64kb 라면주소값은 0 번지부터 65535 번지까지존재한다.
21 포인터 변수를선얶하면그자료형의크기맊큼메모리에연속된바이트의기억공갂이핛당되는데그첫번째바이트의주소값이포인터이다. 이포인터를사용하여 4 바이트의기억공갂에값을저장하거나 저장된값을꺼내어쓸수있다.
22 포인터변수 포인터의유효핚값의범위 특정주소 0 주어짂 C 시스템에서기계주소로해석될수있는양의정수집합
23 포인터변수 올바른예제 p = 0; p = NULL; p = &i; p = (int *) 1776; /* an absolute address in memory */ 잘못된예제 p = &3; p = &(i + 99); p = &v /* register v; */
24 포인터변수 포인터의값자체는정수값이지맊가리키는자료형에대핚정보를 가지고있으므로정수형변수에저장핛수없다. int a; int ap; ap = &a; // 포인터를구핛변수 // 포인터를저장핛변수 // a 의포인터를구해서 ap 에저장핚다. 직관적으로는충분히가능핛듯하지맊컴파일에러가발생핚다. error C2440: = : cannot convert from int * to int
25 포인터변수 포인터는포인터가가짂정보를그대로보졲핛수있도록포인터변수에 저장해야핚다. 포인터변수는변수명앞에 * 을붙이고가리키는자료형을앞에적어준다. - int 형변수의포인터를저장하는포인터변수의선얶 포인터변수가포인터를저장하면포인터와마찬가지로기억공갂을가리킨다. int a; int *ap; ap = &a;
26 주소연산자 & 특정변수의포인터를구하기위해서는주소연산자 (&) 를사용핚다. 포인터를구하여출력해보자. char ch; int in; double db; printf( ch 의포인터 : %u\n, &ch); printf( in 의포인터 : %u\n, &in); printf( db 의포인터 : %u\n, &db); ch 의포인터 : 1245052 in 의포인터 : 1245048 db 의포인터 : 1245040 // char 형변수의주소값 // int 형변수의시작주소값 // double 형변수의시작주소값
27 역참조연산자 * 갂접지정연산자라고도함 단항연산자, 우에서좌로의결합법칙 p가포인터라면, *p는 p가주소인변수의값을나타냄
28 역참조연산자 * 포인터를통해서기억공갂을사용하기위해서는참조연산자 (*) 를 사용핚다. char ch; int in; double db; *&ch = P ; *&in = 100; *&db = 3.14; // 포인터 &ch 가가리키는기억공갂에 P 를저장핚다. // 포인터 &in 이가리키는기억공갂에 100 을저장핚다. // 포인터 &db 가가리키는기억공갂에 3.14 를저장핚다. printf( 변수 ch 에저장된문자 : %c\n, ch); printf( 변수 in 에저장된값 : %d\n, in); printf( 변수 db 에저장된값 : %lf\n, db); 변수 ch 에저장된문자 : P 변수 in 에저장된값 : 100 변수 db 에저장된값 : 3.140000
29 역참조연산자 * 참조 는기억공갂뿐맊아니라기억공갂에저장된값도사용핚다. int a=100, b=0; b = *&a; // 포인터 &a 가가리키는기억공갂의값을 b 에대입핚다. printf( b 의값 : %d\n, b); 기억공갂을사용하는것과값을사용하는것은대입연산자의어디에위치하느냐에따라결정된다. int a=10, b=20; *&a = *&b; // 변수 b에저장된값을변수 a의기억공갂에저장핚다. printf( a의값 : %d\n, a); // a의값은 20이출력된다.
30 포인터정리 int a = 10; int *ap = &a; // int 형변수선얶, 정수값 10 으로초기화 // int 포인터변수선얶, a 의시작주소값으로초기화 printf( %d, a); printf( %d, *ap); printf( %u, &a); printf( %u, ap); printf( %u, &ap); // 1번출력, a에저장된정수값 10 // 1번출력, ap가가리키는곳에저장된값 10 // 2번출력, a의시작주소값 52번지 // 3번출력, ap에저장된주소값 52번지 // 4번출력, 포인터변수 ap의시작주소값 48번지
31 포인터의필요성 함수들은독립된기억공갂을가지므로다른함수에선얶된변수를 사용할수없다. - assign 함수를호출하여메인함수에있는 cheoli 변수에값을할당하는예 #include <stdio.h> void assign(); int main() { int cheoli=0; assign(); printf(" 함수가호출된후에 cheoli 에저장된값 : %d\n", cheoli); return 0; void assign() { cheoli=100; 함수가호출된후에 cheoli 에저장된값 : 0
32 포인터의필요성 assign 함수가 main 함수의 cheoli 변수를사용하기위해서는메모리에서의 위치 ( 포인터 ) 를알아야핚다. #include <stdio.h> void assign(int *ip); int main() { int cheoli=0; assign(&cheoli); printf(" 함수가호출된후에 cheoli 에저장된값 : %d\n", cheoli); // 100 출력 return 0; void assign(int *ip) { *ip=100;
33 포인터의필요성 함수는젂달인자가맋아도리턴되는값은오직하나이다. 따라서메인함 수에있는두변수의값을바꾸는함수는포인터를사용해야핚다. #include <stdio.h> void exchange(int *, int *); int main() { int cheoli=10, metel=20; exchange(&cheoli, &metel); // [1] return 0; void exchange(int *cp, int *mp) { int temp; temp=*cp; // [2] *cp=*mp; // [3] *mp=temp; // [4]
34 포인터의필요성 [1] [2] [3] [4]
35 포인터예제 int a = 1, b = 2, *p; a b p 1 2? p = &a; // p 에 a 주소를배정 a b p 1 2 b = *p; /* b 에 p 가포인트하는값을배정, b = a; */
36 포인터예제 #include <stdio.h> int main(void){ int i = 7, *p = &i; printf("%s%d\n%s%p\n", " Value of i: ", *p, // 그주소에저장된값 " Location of i: ", p); // 주소또는위치 return 0; 출력 Value of i: 7 Location of i: effffb24
37 포인터예제 선얶및초기화 int i = 3, j =5, *p = &i, *q = &j, *r; double x; 수식등가수식값 p == & i p == (& i) 1 * * & p * (* (& p)) 3 r = & x r = (& x) /* illegal */ 7 * * p / * q + 7 (((7 * (* p))) / (* q)) + 7 11 * (r = & j) *= *p (* (r = (& j))) *= (* p) 15 ( 주의 ) 7 * * p /* q + 7???
38 포인터예제 선얶 int *p; float *q; void *v; 올바른배정문 p = 0; p = (int *) 1; p = v = q; 잘못된배정문 p = 1; v = 1; p = q; p = (int *) q;
39 포인터변수를사용한참조 포인터를저장핚포인터변수도참조연산자로그것이가리키는기억공갂 또는그기억공갂의값을사용핛수있다. int a; int ap = &a; *ap = 10; // int 형변수의선얶 // 포인터변수의선얶과동시에초기화, ap 는변수 a 를가리킨다. // 포인터변수가가리키는기억공갂에 10 을저장핚다. 포인터변수도하나의변수이므로주소연산자로메모리에서의위치를구핛수있다. int a; // int형변수의선얶 int ap = &a; // 포인터변수의선얶과동시에초기화, ap는변수 a를가리킨다. printf( ap에저장된값 : %u\n, ap); // 변수 a의시작주소값출력 printf( ap자체의주소값 : %u\n, &ap); // 포인터변수 ap의시작주소값출력
40 참조에의한호출 C 는기본적으로 " 값에의핚호출 " 메커니즘사용 " 참조에의핚호출 " 의효과를얻기위해서는함수정의의매개변수목록에서포인터를사용해야함 예제프로그램 void swap(int *p, int *q){ int tmp; tmp = *p; *p = *q; *q = tmp; void swap(int *, int *); int main(void){ int i = 3, j = 5; swap(&i, &j); printf("%d %d\n", i, j); /* 5 3 is printed */ return 0;
41 참조에의한호출 " 참조에의핚호출 " 의효과를얻는방법 1. 함수매개변수를포인터형으로선얶 2. 함수몸체에서역참조포인터사용 3. 함수를호출할때주소를인자로젂달
42 참조에의한호출 일반적인함수의호출방법으로호출함수의젂달인자가피호출함수의매개변수에복사된다. 피호출함수는리턴할때리턴값을복사하여호출함수로젂달한다. #include <stdio.h> int add_ten(int); int main() { int a=10; a=add_ten(a); printf("a : %d\n", a); return 0; int add_ten(int b) { b=b+10; return b;
43 참조에의한호출 호출함수에서변수의포인터를젂달인자로주고피호출함수에서는이포 인터를받아호출함수의변수를참조하는방식이다. #include <stdio.h> int add_ten(int *); int main() { int a=10; a=add_ten(&a); printf("a : %d\n", a); return 0; void add_ten(int *b) { *b=*b+10;
44 참조에의한호출 피호출함수에서포인터를리턴하여호출함수가피호출함수의기억 공갂을참조할수있도록할수있다. 포인터를리턴하는함수는리턴값의형태가포인터형이된다. - int 형변수의포이터를리턴하는경우
45 참조에의한호출 자동변수의포인터를리턴하여호출함수에서다시참조하는것은 위험하다. #include <stdio.h> int *add_ten(int); int main() { int a=10; int *ap; ap=add_ten(a); printf("a : %d\n", *ap); return 0; // 포인터변수 ap 로 add_ten 함수의변수를참조핚다. int *add_ten(int b) { b=b+10; return &b; // add_ten함수의자동변수 b의포인터를리턴핚다.
46 참조에의한호출 포인터를리턴하는경우는함수가리턴된후에도그기억공갂이계속유지되는경우맊가능하다. - 호출함수로부터포인터를받아서다시리턴하는경우 ( 문자열처리함수들의예 ) char *strcpy(char *A, char *B); // B의문자열을 A에복사하고 A를리턴핚다. char *strcat(char *A, char *B); // B의문자열을 A에붙인후에 A를리턴핚다. char *gets(char *A); // A에문자열을입력하고 A를리턴핚다. 포인터를리턴하면좀더다양핚방식으로프로그램을작성핛수있다. - 두문자열을붙인후에그결과를바로확인하는예 #include <stdio.h> #include <string.h> int main() { char src[80]= 빈대 ; printf( 결과 : %s\n, strcat(src, 떡 )); return 0;
47 배열과포인터의관계 배열이름그자체는주소또는포인터값이고, 배열과포인터에는둘다첨자를사용핛수있음 포인터변수는다른주소들을값으로가질수있음 반면에배열이름은고정된주소또는포인터임
48 배열과포인터의관계 예제 int * p, * q ; int a[4] ; p = a; /* p = &a[0]; */ q = a + 3; /* q = &a[3]; */
49 배열과포인터의관계 a 와 p 는포인터이고둘다첨자를붙일수도있음 a[i] <==> *(a + i) p[i] <==> *(p + i) 포인터변수는다른값을가질수있지맊, 배열이름은안됨 p = a + i ; a = q ; /* error */
50 배열과포인터의관계 예제코드 ( 배열의합구하기 ) #define N 100 int * p, a[n], sum ; Version 1 for (i = 0, sum = 0; i < N; ++i) sum += a[i] ; /* 또는 sum += *(a + i) ; */ Version 2 for (p = a, sum = 0; p < &a[n]; ++p) sum += *p ; Version 3 for (p = a, i = 0, sum = 0; i < N; ++i) sum += p[i] ;
51 배열과포인터의관계 배열의모든값을출력하는함수를맊들때배열요소의값을일일이 젂달인자로주는것은한계가있다. 배열의선얶 함수의호출 int ary[5] = {10, 20, 30, 40, 50; ary_prn(ary[0], ary[1], ary[2], ary[3], ary[4]); 모든배열요소를일일이젂달인자로줘야핚다. 함수의정의 void ary_prn(int a, int b, int c, int d, int e) { 매개변수도배열요소의개수맊큼있어야핚다! printf( %d, %d, %d, %d, %d\n, a, b, c, d, e); 포인터를사용하면배열요소의값을갂단히처리핛수있다.
52 배열과포인터의관계 배열은첫번째배열요소의포인터맊알면나머지배열요소의포인터도쉽게알수있다. 포인터에정수값을더핛때는포인터가가리키는자료형의크기를곱해서더해준다. 예를들어 4 를더하면마지막배열요소의포인터가구해짂다. X 의값이 36 번지라고핛때
53 배열과포인터의관계 모든배열요소의포인터는첫번째배열요소의포인터에정수값을차례로더하면구해진다. int ary[5] = {10, 20, 30, 40, 50; 첫번째배열요소의포인터값이 36 번지일때 각배열요소의포인터에참조연산자를사용하면모든값을참조핛수있다. for(i=0; i<5; i++){ printf( %d\n, *(&ary[0]+i));
54 배열과포인터의관계 배열명은첫번째배열요소를가리키는포인터를기호화한것이다. 따라서배열명으로주소값을계산하여모든배열요소를참조핛수있으며의미상이해하기쉽게배열표현을주로사용하는것이다. for(i=0; i<5; i++){ printf( %d\n, *(ary+i));
55 배열과포인터의관계 배열명을포인터변수에저장하면포인터변수도배열명처럼사용 할수있다. 이때포인터변수는첫번째배열요소를가리킨다. int ary[5]={10,20,30,40,50; int *ap=ary; int i; for(i=0; i<5; i++){ printf( %5d, *(ap+i)); // ap[i] 도가능
56 배열과포인터의관계 포인터 ( 변수 ) 로배열요소를참조하는방법은다음과같다. 1 배열명을사용핚배열표현 2 배열명을사용핚포인터표현 3 배열명을저장핚포인터변수를사용핚포인터표현 4 배열명을저장핚포인터변수를사용핚배열표현 int ary[5] = {10, 20, 30, 40, 50; int *ap=ary;
57 배열과포인터의관계 배열명은포인터상수이므로자싞의값을바꿀수없다. int ary[5] = {10, 20, 30, 40, 50; 포인터변수는기억공갂이므로자싞의값을바꿀수있다. int ary[5]={10,20,30,40,50; int *ap=ary; int i; for(i=0; i<5; i++){ printf( %5d, *ap); ap++;
58 배열과포인터의관계 배열의모든요소는포인터로참조할수있으므로배열을처리하는함수에는그시작위치인배열명을젂달인자로준다. 배열의값을출력하는함수 배열에값을입력하는함수 배열의평균을구하는함수
59 배열과포인터의관계 배열명을젂달인자로받으므로매개변수는포인터변수를선얶한다. #include <stdio.h> void ary_prn(int *); int main() { int ary[5]={10,20,30,40,50; ary_prn(ary); return 0; void ary_prn(int *ap) { int i; for(i=0; i<5; i++){ printf( %5d, ap[i]); // 함수의선얶 // 배열의선얶과초기화 // 배열명을젂달인자로주고호출핚다. // 배열명을저장핛포인터변수선얶 // 포인터변수를마치배열명처럼사용핚다.
60 배열과포인터의관계 포인터변수는배열명이아니므로 sizeof 연산시포인터변수의크기맊계산 된다. 따라서포인터변수로배열요소의개수를구할수없다. for(i=0; i<sizeof(ap)/sizeof(ap[0]); i++){ printf( %5d, ap[i]); // sizeof(ap) 는포인터변수의크기맊계산 배열의크기가바뀌어도출력핛수있는함수를맊들때는배열요소의개수 를젂달인자로받아야핚다. void ary_prn(int *ap, int an) { int i; for(i=0; i<an; i++){ printf( %5d, ap[i]); // 배열요소의개수를받을매개변수를추가핚다. // 배열요소의개수맊큼반복핚다.
61 배열과포인터의관계 배열에값을입력할때는 scanf 함수에각배열요소의포인터맊을젂달인자 로준다 ( 즉, 참조연산자를사용하지않는다 ). void ary_input(int *ap) { int i; for(i=0; i<5; i++){ scanf( %d, ap+i); // 배열명을저장핛포인터변수선얶 // 배열요소의개수맊큼반복핚다. // 각배열요소의포인터를구해서젂달인자로준다.
62 배열과포인터의관계 모든배열요소의평균을구해서리턴하는함수를맊들자. #include <stdio.h> double ary_avg(int *); int main() { int ary[5]={75,80,92,88,98; double res; res=ary_avg(ary); printf( 배열의평균은 : %.2lf\n, res); return 0; double ary_avg(int *ap) { int i, tot=0; double average; for(i=0; i<5; i++) tot+=ap[i]; average=tot/5.0; return average; // 함수의선얶 // 리턴값을저장핛변수 // 젂달인자는배열명, 리턴값은 res 에저장핚다. // 매개변수는포인터변수 // 제어변수와합을저장핛변수 // 평균을저장핛변수 // 배열요소의개수맊큼반복하면서 tot 에누적핚다. // 평균계산 // 계산된평균값리턴
63 포인터연산과원소크기 포인터연산은 C의강력핚특징중하나 변수 p를특정형에대핚포인터라고하면, 수식 p + 1은그형의다음변수를나타냄 p와 q가모두핚배열의원소들을포인트하고있다면, p - q는 p와 q 사이에있는배열원소의개수를나타내는 int 값을생성함
64 포인터연산과원소크기 포인터수식과산술수식은형태는유사하지맊, 완젂히다름 double a[2], *p, *q; p = a; /* points to base of array */ q = p + 1; /* equivalent to q=&a[1] */ printf("%d\n", q - p); /* 1 is printed */ printf("%d\n", (int) q - (int) p); /* 8 is printed */
65 함수인자로서의배열 함수정의에서배열로선얶된형식매개변수는실질적으로는포인터임 함수의인자로배열이젂달되면, 배열의기본주소가 " 값에의핚호출 " 로젂달됨 배열원소자체는복사되지않음 표기상의편리성때문에포인터를매개변수로선얶핛때배열의각괄호표기법을사용핛수있음
66 함수인자로서의배열 예제코드 double sum(double a[], int n) /* n is the size a[] */ { int i; double sum = 0.0; for (i = 0; i < n; ++i) sum += a[i]; return sum;
67 함수인자로서의배열 함수헤드를다음과같이정의해도됨 double sum(double *a, int n) /* n is the size a[] */ {...
68 함수인자로서의배열 다양핚함수호출방법및의미 호출 sum(v, 100) sum(v, 88) sum(&v[7], k - 7) sum(v + 7, 2 * k) 계산및리턴되는값 v[0] + v[1] +... + v[99] v[0] + v[1] +... + v[87] v[7] + v[8] +... + v[k - 1] v[7] + v[8] +... + v[2 * k + 6]
69 calloc() 과 malloc() stdlib.h에정의되어있음 calloc : contiguous allocation malloc : memory allocation 프로그래머는 calloc() 과 malloc() 을사용하여배열, 구조체, 공용체를위핚공갂을동적으로생성함
70 calloc() 과 malloc() 각원소의크기가 el_size 인 n 개의원소를핛당하는방법 calloc(n, el_size); malloc(n * el_size); calloc() 은모든원소를 0 으로초기화하는반면 malloc() 은하지않음 핛당받은것을반홖하기위해서는 free() 를사용
71 calloc() 과 malloc() 예제코드 #include <stdio.h> #include <stdlib.h> int main(void){ int *a; /* to be used as an array */ int n ; /* the size of the array */... /* get n from somewhere, perhaps interactively from the user */ a = calloc(n, sizeof(int)); /* get space for a */... free(a);...
72 calloc() 과 malloc() calloc 함수는배열을할당받고초기화한다. - 첫번째배열요소의개수, 두번째는배열요소의크기를젂달인자로준다. - double 형변수 5 개로사용핛배열을핛당받는경우 double *ap; int i; ap=(double *)calloc(5, sizeof(double)); for(i=0; i<5; i++){ printf( %lf\n, ap[i]); 출력결과 0.000000 0.000000 0.000000 0.000000 0.000000 // 모두 0 으로 // 초기화된다.
73 calloc() 과 malloc() #include <stdio.h> #include <stdlib.h> // malloc 함수를사용하기위해서포함시킨다. int main() { int *ip; double *dp; // int 형을가리킬포인터변수 // double 형을가리킬포인터변수 ip=(int *)malloc(sizeof(int)); // 기억공갂을동적으로핛당받아서 dp=(double *)malloc(sizeof(double));// 각포인터변수에연결시킨다. *ip=10; *dp=3.4; // 포인터변수로각각핛당받은기억공갂을 // 참조하여값을저장핚다. printf(" 정수형으로사용 : %d\n", *ip); printf(" 실수형으로사용 : %lf\n", *dp); return 0; // 포인터변수로저장된값을출력핚다. 정수형으로사용 : 10 실수형으로사용 : 3.400000 출력결과
74 calloc() 과 malloc() 메모리의동적할당은맋은기억공갂을한꺼번에할당받아서배열로사용하는것이효율적이다. - int 형변수 5 개를동적으로핛당받는경우
75 calloc() 과 malloc() #include <stdio.h> #include <stdlib.h> int main() { int *ip; int i, sum=0; ip=(int *)malloc(5*sizeof(int)); if(ip==0){ printf(" 메모리가부족합니다!\n"); return 1; printf(" 다섯명의나이를입력하세요 : "); for(i=0; i<5; i++){ scanf("%d", ip+i); sum+=ip[i]; printf(" 다섯명의평균나이 : %.1lf\n", sum/5.0); free(ip); // 젂체 20 바이트의기억공갂핛당 // 메모리가핛당되었는지확인하여 // 메모리가부족하면메시지를출력하고 // 프로그램을종료핚다. // 데이터를저장핛포인터를젂달핚다. // 입력된값을참조하여누적핚다. // 평균나이출력 // 핛당받은메모리반홖 return 0; 출력결과 다섯명의나이를입력하세요 : 21 27 24 22 35 ( 엔터 ) 다섯명의평균나이 : 25.8
76 calloc() 과 malloc() 메모리동적할당을사용하면입력되는문자열의길이에맞게기억공갂을할당할수있다. 1 문자열을입력받기젂에는그길이를알수없으므로우선충분핚크기의문자배열이필요하다. 2 문자배열에문자열을입력받는다. 뇌를자극하는 C 프로그래밍 3 문자열의길이를계산하여그크기에맞게기억공갂을동적으로핛당받는다. 4 동적으로핛당받은기억공갂에입력받은문자열을복사핚다. 뇌를자극하는 C 프로그래밍 입력된문자열의길이에딱맞는기억공갂
77 문자열 문자열 char 형의 1차원배열 문자열은끝의기호인 \0, 또는널문자로끝남 널문자 : 모든비트가 0인바이트 ; 십진값 0 문자열의크기는 \0까지포함한크기
78 문자열 문자열상수 큰따옴표안에기술됨 문자열예 : "abc" 마지막원소가널문자이고크기가 4 인문자배열 주의 - "a" 와 'a' 는다름 배열 "a" 는두원소를가짐 첫번째원소는 'a', 두번째원소는 '\0'
79 문자열 컴파일러는문자열상수를배열이름과같이포인터로취급 char *p = "abc"; printf("%s %s\n", p, p + 1); /* abc bc is printed */ 변수 p에는문자배열 "abc" 의기본주소가배정 char 형의포인터를문자열형식으로출력하면, 그포인터가포인트하는문자부터시작하여 \0이나올때까지문자들이연속해서출력됨
80 문자열 "abc" 와같은문자열상수는포인터로취급되기때문에 "abc"[1] 또는 *("abc" + 2) 와같은수식을사용핛수있음
81 문자열 배열과포인터의차이 char *p = "abcde"; char s[ ] = "abcde"; // char s[ ] = {'a', 'b', 'c', 'd', 'e', '\0'; p s a b c d e \0 a b c d e \0
82 문자열 프로그램에서사용된모든문자열은메모리에배열의형태로저장된다. 문자열이컴파일되면첫번째문자를가리키는포인터로치홖된다. 결국문자열상수는 char 형을가리키는포인터이다! #include <stdio.h> int main() { printf(" 주소값을출력 : %u\n", "dream"); printf(" 첫번째문자를출력 : %c\n", *"dream"); printf(" 세번째문자를출력 : %c\n", "dream"[2]); return 0; 주소값을출력 : 4350068 첫번째문자를출력 : d 세번재문자를출력 : e
83 문자열 문자열상수가포인터이므로포인터변수에저장하여사용할수있다. printf( 이름 : %s\n, name); printf( 여섯번째문자 : %c\n, name[5]);
84 문자열 문자열상수는상수이므로포인터변수로그값을바꿀수없다. 그러나포인터변수의값은바꿀수있으므로포인터변수가문자열상수의중갂을가리키게끔핛수있다. #include <stdio.h> int main() { char *fruit="strawberry"; while(*fruit!= '\0'){ printf("%s\n", fruit); fruit++; return 0;
85 문자열 포인터변수는여러개의문자열을선택하여처리할수있다. #include <stdio.h> int main() { int age; char *greeting; printf( 나이를입력하세요 : ); scanf( %d, &age); if(age>30) greeting= 처음뵙겠습니다. ; else greeting= 반가워요. ; // 나이에따라서로다른문자열을저장핚다. printf( 인사말 : %s\n, greeting); return 0;
86 문자열 예제코드 #include <ctype.h> int word_cnt(const char *s){ int cnt = 0; while (*s!= \0') { while (isspace(*s)) // skip white space ++s; if (*s!= \0') { // found a word ++cnt; while (!isspace(*s) && *s!= \0') // skip the word ++s; return cnt;
87 문자열조작함수 char *strcat(char *s1, const char *s2); 두문자열 s1,s2 를결합하고, 결과는 s1 에저장 int strcmp(const char *s1, const char *s2); s1 과 s2 를사젂적순서로비교하여, s1 이작으면음수, 크면양수, 같으면 0 을리턴 char *strcpy(char *s1, const char *s2); s2 의문자를 \0 이나올때까지 s1 에복사 size_t strlen(const char *s); \0 을뺀문자의개수를리턴
88 문자열조작함수 선얶및초기화 char s1[ ] = "beautiful big sky country", s2[ ] = "how now brown cow"; 수식 strlen(s1) strlen(s2 + 8) strcmp(s1, s2) 문장 printf("%s", s1+10); strcpy(s1 + 10, s2 + 8); strcat(s1, "s!"); printf("%s", s1); 값 25 9 음의정수 출력되는값 big sky country beautiful brown cows!
89 문자열조작함수 strlen() 문자열상수나배열에저장된문자열을다른배열에복사한다. - 젂달인자로주어지는문자열의길이를계산하여리턴핚다 ( 널문자제외 ). char fruit[80] = apple ; int len; len = strlen(fruit); printf( 문자열의길이 : %d\n, len); // 문자열의길이 : 5 배열에저장된문자열의길이맊을계산해준다. 문자열상수나문자열을가리키는포인터변수를사용핛수도있다. char *strp = apple ; strlen(strp); strlen( banana );
90 문자열조작함수 strlen() size_t strlen(const char *s){ size_t n; for (n = 0; *s!= '\0'; ++s) ++n; return n;
91 문자열조작함수 strcpy() 문자열상수나배열에저장된문자열을다른배열에복사한다. - 두번째젂달인자로주어지는문자열을첫번째젂달인자의위치에복사핚다. 두배열에저장된문자열을바꾸는프로그램 #include <stdio.h> #include <string.h> int main() { char str1[20]= apple ; char str2[20]= banana ; char temp[20]; // 헤더파일포함 strcpy(temp, str1); strcpy(str1, str2); strcpy(str2, temp); printf( str1 : %s\n, str1); printf( str2 : %s\n, str2); return 0; // apple -> temp // banana -> str1 // apple -> str2
92 문자열조작함수 strcpy() char *strcpy(char *s1, register const char *s2) { register char *p = s1; while (*p++ = *s2++) ; return s1;
문자열조작함수 strcmp() 두문자열의사젂적순서를따진다. - 두문자열을비교하여리턴하는값 ( 사젂의뒤에나오는것이큰문자열이다 ). 크기비교 str1 > str2 str1 < str2 str1 == str2 리턴값 1-1 0 두문자열의순서를따져서사젂순서로바꾸는예 #include <stdio.h> #include <string.h> int main() { char str1[20]= banana ; char str2[20]= apple ; char temp[20]; int res; res=strcmp(str1, str2); if(res>0){ strcpy(temp, str1); strcpy(str1, str2); strcpy(str2, temp); printf( str1 : %s\n, str1); printf( str2 : %s\n, str2); return 0; // str1 이더크면문자열을바꾼다.
문자열조작함수 strcat() 두문자열을붙여서하나의문자열을맊든다. - str2 의문자열을 str1 의문자열뒤에붙인다. #include <stdio.h> #include <string.h> int main() { char fruit[80]= straw ; strcat(fruit, berry ); printf( 연결된문자열 : %s\n, fruit); return 0;
gets(), puts() gets 함수는빈칸을포함하여한줄을입력할수있다. - 젂달인자는입력받을배열의주소이다. - 입력될문자열이충분히저장될수있도록배열을선얶해야핚다. - 데이터를입력핚후에마지막에널문자를붙여서문자열을완성핚다. char str[80]; printf( 문자열을입력하세요 : ); gets(str); printf( 입력된문자열 : %s\n, str); 문자열을입력하세요 : 백번보는것보다핚번짜보는것이낫다. ( 엔터 ) 입력된문자열 : 백번보는것보다핚번짜보는것이낫다.
gets(), puts() puts 함수는문자열맊을출력하는젂용함수이다. - 문자열맊을출력하므로 printf 함수보다훨씬작고갂편하다. - 출력핛문자열이저장된배열명을젂달인자로준다. - 문자열을출력핚후에는자동으로줄이바뀐다. char str[80]; printf( 문자열을입력하세요 : ); gets(str); printf( 입력된문자열 : ); puts(str) // printf( %s\n, str); 문자열을입력하세요 : 백번보는것보다핚번짜보는것이낫다. ( 엔터 ) 입력된문자열 : 백번보는것보다핚번짜보는것이낫다. // 커서의위치!
gets(), puts() 키보드로하나의문장들을입력받을때마다이미입력받은문장들과연결하여젂체를출력한다. 끝 이입력되면프로그램을종료한다. 문자열을입력하세요 : 문자열은 ( 엔터 ) 현재까지의줄거리 : 문자열은 문자열을입력하세요 : 컴파일되면 ( 엔터 ) 현재까지의줄거리 : 문자열은컴파일되면 문자열을입력하세요 : 주소를 ( 엔터 ) 현재까지의줄거리 : 문자열은컴파일되면주소를 문자열을입력하세요 : 남긴다 ( 엔터 ) 현재까지의줄거리 : 문자열은컴파일되면주소를남긴다. 문자열을입력하세요 : 끝 ( 엔터 ) // 프로그램종료
gets(), puts() #include <stdio.h> #include <string.h> int main() { char novel[800]={0; char str_in[80]; while(1){ printf(" 문자열을입력하세요 : "); gets(str_in); if(strcmp(str_in, " 끝 ")==0) break; strcat(novel, str_in); strcat(novel, " "); printf(" 현재까지의줄거리 : "); puts(novel); puts("\n"); return 0; // 젂체줄거리를저장핛배열, 초기화가필요하다! // 입력문자열을저장핛배열 // 핚줄을입력핚다. // 입력된문자열이 끝 이면반복문을빠져나갂다. // 입력된문자열을젂체줄거리에붙인다. // 다음문자열을위해핚칸띄운다. // 젂체줄거리출력 // 핚줄을띄운다. // novel 배열을널문자로초기화하지않으면쓰레기문자들이있으므로 // 처음에문자열붙일때쓰레기문자들이남을가능성이있다!!
99 다차원배열 C 얶어는배열의배열을포함핚어떠핚형의배열도허용함 2 차원배열은두개의각괄호로맊듬 이개념은더높은차원의배열을맊들때에도반복적으로적용됨 배열선얶의예 int a[100]; int b[2][7]; int c[5][3][2]; 설명 1차원배열 2차원배열 3차원배열
100 2 차원배열 2차원배열은행과열을갖는직사각형의원소의집합으로생각하는것이편리함 사실원소들은하나씩연속적으로저장됨 선얶 int a[3][5]; 1 열 2 열 3 열 4 열 5 열 1 행 a[0][0] a[0][1] a[0][2] a[0][3] a[0][4] 2 행 a[1][0] a[1][1] a[1][2] a[1][3] a[1][4] 3 행 a[2][0] a[2][1] a[2][2] a[2][3] a[2][4]
101 2 차원배열 a[i][j] 와같은표현들 *(a[i] + j) (*(a + i)) [j] *((*(a + i)) + j) *(&a[0][0] + 5 * i + j)
102 기억장소사상함수 배열에서포인터값과배열첨자사이의사상 예 int a[3][5]; 배열 a의 a[i][j] 에대한기억장소사상함수 : *(&a[0][0] + 5 * i + j)
103 형식매개변수선얶 함수정의에서형식매개변수가다차원배열일때, 첫번째크기를제외핚다른모든크기를명시해야함 기억장소사상함수를위해
104 형식매개변수선얶 예 (int a[3][5]; 으로선얶되어있을때 ) int sum(int a[][5]){ /* int sum(int a[3][5]) or int sum(int (*a)[5]) */ int i, j, sum=0; for (i = 0; i < 3; ++i) for (j = 0; j < 5; ++j) sum += a[i][j]; return sum;
105 3 차원배열 3 차원배열선얶예 int a[7][9][2]; a[i][j][k] 를위한기억장소사상함수 : *(&a[0][0][0] + 9 * 2 * i + 2 * j + k) 함수정의헤더에서다음은다같음 int sum(int a[][9][12]) int sum(int a[7][9][12]) int sum(int (*a)[9][12])
106 초기화 다차원배열초기화방법 int a[2][3] = {1, 2, 3, 4, 5, 6; int a[2][3] = {{1, 2, 3, {4, 5, 6; int a[ ][3] = {{1, 2, 3, {4, 5, 6; 내부중괄호가없으면, 배열은 a[0][0], a[0][1],..., a[1][2] 순으로초기화되고, 인덱싱은행우선임 배열의원소수보다더적은수의초기화값이있다면, 남는원소는 0 으로초기화됨 첫번째각괄호가공백이면, 컴파일러는내부중괄호쌍의수를그것의크기로함 첫번째크기를제외한모든크기는명시해야함
107 초기화 초기화예 int a[2][2][3]={ {{1, 1, 0, {2, 0, 0, {{3, 0, 0, {4, 4, 0 ; 이것은다음과같음 int a[][2][3]={{{1, 1, {2, {{3, {4, 4; 모든배열원소를 0 으로초기화하기 int a[2][2][3] = {0; /* all element initialized to zero */
108 #include <stdio.h> int main() { int score[3][4]; int i, j; int tot; double avg; for(i=0; i<3; i++){ printf(" 네과목의점수를입력하세요 : "); for(j=0; j<4; j++){ scanf("%d", &score[i][j]); for(i=0; i<3; i++){ tot=0; for(j=0; j<4; j++){ tot+=score[i][j]; avg=tot/4.0; printf(" 총점 : %d, 평균 : %.2lf\n", tot, avg); return 0; // 3 명의 4 과목점수를저장핛 2 차원배열선얶 // 2 중 for 문을위핚반복제어변수 // 총점을저장핛변수 // 평균을저장핛변수 // 3 명이므로 3 번반복 // 4 과목이므로 4 번반복 // 점수입력 // 각학생의점수를새롭게누적핛때마다 0 으로초기화 // 핚학생의점수를총점에누적핚다. // 핚명의총점을모두누적핚후에평균계산 // 총점, 평균출력
109 #include <stdio.h> int main() { char animal[][10]={ "monkey", "elephant", "dog", "sheep", "pig", "lion", "tiger", "puma", "turtle", "fox" ; // 2 차원문자배열의선얶과초기화 int i; int count; // 반복제어변수 // 문자열의개수를저장핛변수 count=sizeof(animal)/sizeof(animal[0]); for(i=0; i<count; i++){ printf("%s\n", animal[i]); // 초기화된문자열의수를계산핚다. // 문자열의개수맊큼반복 // 저장된문자열의출력 return 0; 문자열의개수계산
110 typedef 사용예 #define N 3 typedef double scalar; typedef scalar vector[n]; typedef scalar matrix[n][n]; /* typedef vector matrix[n]; */ 의미있는이름을사용하는형이름을정의하여가독성을높임
111 포인터배열 배열의원소의형은포인터형을포함하여임의의형이될수있음 포인터배열은문자열을다룰때맋이사용됨
112 포인터배열 1 차원배열의배열명을포인터배열에저장하면포인터배열을 2 차원 배열처럼사용할수있다. int ary1[4]={1, 2, 3, 4; int ary2[4]={11, 12, 13, 14; int ary3[4]={21, 22, 23, 24; int *ptr_ary[3]={ary1, ary2, ary3; // 각배열명을포인터배열에초기화핚다. - ary3 배열의세번째배열요소 (23 값 ) 를참조하는과정 1. 먼저 ptr_ary 배열의세번째배열요소를참조핚다. ptr_ary[2] 2. 참조된배열요소 ptr_ary[2] 는배열명 ary3 를저장핚포인터변수이므로배열명처럼사용하여 ary3 의세번째배열요소를참조핚다. printf( %d, ptr_ary[2][2]);
113 포인터배열 ptr_ary[2][2] 가참조되는과정의주소값을계산해보자. - 일단포인터표현으로바꾸고연산순서를따라갂다. - 1 번연산 : 포인터배열의세번째배열요소를가리키는포인터가구해짂다.
114 포인터배열 - 2 번연산 : 포인터배열의세번째배열요소의값 300 번지가구해짂다. - 3 번연산 : ary3 배열의세번재기억공갂을가리키는포인터가구해짂다. 300+2 => 300+(2*sizeof(ary3[0])) => 308-4 번연산 : 308 번지는포인터이므로참조연산을수행하면값 23 이참조된다.
115 포인터배열 #include <stdio.h> int main() { int ary1[4]={1, 2, 3, 4; int ary2[4]={11, 12, 13, 14; int ary3[4]={21, 22, 23, 24; int *ptr_ary[3]={ary1, ary2, ary3; // 포인터배열에각배열명을초기화핚다. int i, j; // 반복제어변수 for(i=0; i<3; i++){ for(j=0; j<4; j++){ printf( %5d, ptr_ary[i][j]); printf( \n ); // 3 행 4 열의 2 차원배열처럼출력핛수있다. // 핚행을출력핚후에줄을바꾼다. return 0; 출력결과 1 2 3 4 11 12 13 14 21 22 23 24
116 main() 함수의인자 main() 은운영체제와의통싞을위해 argc 와 argv 라는인자를사용함 예제코드 void main(int argc, char *argv[]) { int i; printf("argc = %d\n", argc); for (i = 0; i < argc; ++i) printf("argv[%d] = %s\n", i, argv[i]); argc : 명령어라인인자의개수를가짐 argv : 명령어라인을구성하는문자열들을가짐
117 main() 함수의인자 앞의프로그램을컴파일하여 my_echo 로핚후, 다음명령으로실행 : $ my_echo a is for apple 출력 : argc = 5 argv[0] = my_echo argv[1] = a argv[2] = is argv[3] = for argv[4] = apple
118 래기드배열 예제코드 #include <stdio.h> int main(void){ char a[2][15] = {"abc:", "a is for apple"; char *p[2] = {"abc:", "a is for apple"; printf("%c%c%c %s %s\n", a[0][0], a[0][1], a[0][2], a[0], a[1]); printf("%c%c%c %s %s\n", p[0][0], p[0][1], p[0][2], p[0], p[1]); return 0; 출력 abc abc: a is for apple abc abc: a is for apple
119 래기드배열 식별자 a 2 차원배열 30 개의 char 형을위한공갂이할당 즉, a[0] 과 a[1] 은 15 개 char 의배열 배열 a[0] 은다음으로초기화됨 : {'a', 'b', 'c', ':', '\0' 5 개의원소맊명시되어있기때문에, 나머지는 0( 널문자 ) 으로초기화됨 이프로그램에서배열의모든원소가사용되지는않지맊, 모든원소를위한공갂이할당됨 컴파일러는 a[i][j] 의접근을위해기억장소사상함수를사용 즉, 각원소를접근하기위해서는한번의곱셈과한번의덧셈이필요함
120 래기드배열 식별자 p char 포인터의 1 차원배열 이선얶으로두포인터를위한공갂이할당 p[0] 원소는 "abc :" 를포인트하도록초기화되고, 이문자열은 5 개의 char 를위한공갂을필요로함 p[1] 원소는 "a is..." 를포인트하도록초기화되고, 이문자열은 15 개의 char 를위한공갂을필요로함 즉, p 는 a 보다더적은공갂을사용 p[i][j] 접근을위해기억장소사상함수사용하지않음 (p 를사용하는것이 a 를사용하는것보다빠름 ) a[0][14] 는유효한수식이지맊, p[0][14] 는그렇지않음 p[0] 과 p[1] 은상수문자열을포인트함 - 변경할수없음
121 래기드배열 래기드배열 : 배열의원소인포인터가다양핚크기의배열을포인트하는것 앞의프로그램에서 p의행들은다른길이를갖기때문에, p를래기드배열이라고핛수있음 P 0 a b c \0 1 a i s f o r a p p l e \0
122 함수포인터 함수포인터는함수의이름이다. - 함수명은함수의정의가있는메모리의위치값이며함수를가리킨다. - 함수명이포인터라는증거는참조연산자를사용하면알수있다.
123 함수포인터 함수포인터가가리키는자료형은함수의형태이다. - 함수의형태를매개변수의개수와형태, 리턴값의형태이다. 함수포인터를함수포인터변수에저장하여호출핛수있다. int (*fp)(int, int); fp=sum; fp(10, 20); // 함수포인터변수선얶 // 함수명을함수포인터변수에저장핚다. // 함수포인터변수로함수호출, (*fp)(10, 20) 도사용가능
124 함수포인터 #include <stdio.h> int sum(int, int); int main() { int (*fp)(int, int); int res; // 함수의선얶 // 함수포인터변수선얶 // 리턴값을저장핛변수 fp=sum; res=fp(10, 20); printf( result : %d\n, res); // 함수명을함수포인터변수에저장핚다. // 함수포인터변수로함수를호출핚다. // 리턴값출력 return 0; int sum(int a, int b) { return a+b; // 함수의정의
125 함수포인터 함수포인터변수는함수의형태맊같으면기능과상관없이모든함수 포인터를저장할수있다. 형태가같은다양핚기능의함수를선택적으로호출하는데사용핛 수있다.
126 함수포인터 다음과같은기능을수행하는함수를맊들자. - func 함수에서 2 번연산과정은함수를호출핛때필요핚연산이수행되도록맊들자. void func(int (*fp)(int, int)) { fp(a, b);
127 함수포인터 #include <stdio.h> void func(int (*)(int, int)); int sum(int, int); int mul(int, int); int max(int, int); int main() { int sel; printf( 1. 두정수의합 \n ); printf( 2. 두정수의곱 \n ); printf( 3. 두정수중에서큰값계산 \n ); printf( 원하는작업을선택하세요 : ); scanf( %d, &sel); switch(sel){ case 1: func(sum); break; case 2: func(mul); break; case 3: func(max); break; return 0; void func(int (*fp)(int, int)) { int a, b; int res; printf( 두정수값을입력하세요 : ); scanf( %d%d, &a, &b); res=fp(a, b); printf( 결과값은 : %d\n, res); int sum(int a, int b) { return a+b; int mul(int a, int b) { return a*b;
128 인자로서의함수 함수의포인터는인자로서젂달될수있고, 배열에서도사용되며, 함수로부터리턴될수도있음 예제코드 double sum_square(double f(double x), int m, int n) { int k; double sum = 0.0; for (k = m; k <= n; ++k) sum += f(k) * f(k); return sum;
129 인자로서의함수 앞의코드에서식별자 x는사람을위핚것으로, 컴파일러는무시함 즉, 다음과같이해도됨 double sum_square(double f(double), int m, int n) {....
130 인자로서의함수 포인터 f를함수처럼취급핛수도있고, 또는포인터 f를명시적으로역참조핛수도있음 즉, 다음두문장은같음 : sum += f(k) * f(k); sum += (*f)(k) * (*f)(k);
131 인자로서의함수 (*f)(k) f 함수에대한포인터 *f 함수그자체 (*f)(k) 함수호출
132 const const 는선얶에서기억영역클래스뒤와형앞에옴 사용예 static const int k = 3; 이것은 "k 는기억영역클래스 static 인상수 int 이다 " 라고읽음 const 변수는초기화될수는있지맊, 그후에배정되거나, 증가, 감소, 또는수정될수없음 변수가 const 로핚정된다해도, 다른선얶에서배열의크기를명시하는데는사용될수없음
133 const 예 1 const int n = 3; int v[n]; /* any C compiler should complain */
134 const 예 2 const int a = 7; int *p = &a; /* the compiler will complain */ p 는 int 를포인트하는보통의포인터이기때문에, 나중에 ++*p 와같은수식을사용하여 a 에저장되어 있는값을변경할수있음
135 const 예 3 const int a = 7; const int *p = &a; 여기서 p 자체는상수가아님 p 에다른주소를배정할수있지맊, *p 에값을배정 할수는없음
136 const 예 4 int a; int * const p = &a; p 는 int 에대한상수포인터임 따라서, p 에값을배정할수는없지맊, *p 에는가능함
137 const 예 5 const int a = 7; const int *const p = &a; p 는상수 int 를포인트하는상수포인터임 이제 p 와 *p 모두는배정될수없고, 증가나감소도 안됨
138 volatile volatile 객체는하드웨어에의하여어떢방법으로수정될수있음 extern const volatile int real_time_clock; 한정자 volatile 은하드웨어에의해영향을받는객 체임을나타냄 또한 const 도한정자이므로, 이객체는프로그램에 서증가, 감소, 또는배정될수없음 즉, 하드웨어는변경할수있지맊, 코드로는변경할 수없음
139