누구나즐기는 C 언어콘서트 제 11 장구조체
이번장에서학습할내용 구조체의개념, 정의, 초기화방법 구조체와포인터와의관계 공용체와 typedef 구조체는서로다른데이터들을하나로묶는중요한도구입니다.
자료형의분류 기본자료형 : char, int, float, double 등 자료형 파생자료형 : 배열, 열거형, 구조체, 공용체
구조체의필요성 학생에대한데이터를하나로모으려면? 학번 : 20100001( 정수 ) 이름 : 최자영 ( 문자열 ) 학점 : 4.3( 실수 ) int number; char name[10]; double grade; 와같이개별변수로나타낼수있지만묶을수가있나?
구조체의필요성 int number; char name[10]; double grade; 구조체를사용하면변수들을하나로묶을수있습니다.
구조체와배열 구조체 vs 배열 같은타입의집합 다른타입의집합
중간점검 1. 구조체와배열의차이점을이야기해보라. 2. 복소수, 날짜, 화면의좌표, 사각형등을표현하는데필요한데이터를나열해보라.
구조체선언 구조체선언형식 struct 태그 { }; 자료형멤버 1; 자료형멤버 2;...
구조체선언 구조체선언은변수선언은아님
구조체선언의예 // x 값과 y 값으로이루어지는화면의좌표 struct point { // 사각형 struct rect { int x; // x 좌표 int x; int y; // y 좌표 int y; }; int width; // 복소수 struct complex { }; int grade; double real; double imag; // 실수부 // 허수부 // 직원 struct employee { }; char name[20]; // 이름 // 날짜 struct date { int month; int day; }; int age; int gender; int salary; // 성별 // 월급 // 나이 int year; };
구조체변수선언 구조체정의와구조체변수선언은다르다.
구조체의초기화 중괄호를이용하여초기값을나열한다. struct student { int number; char name[10]; double grade; }; struct student s1 = { 24, "Kim", 4.3 };
구조체멤버참조 구조체멤버를참조하려면다음과같이. 연산자를사용한다. s1.number = 26; strcpy(s1.name, "Kim"); s1.grade = 4.3; // 정수멤버 // 문자열멤버 // 실수멤버.. 기호는구조체에서멤버를참조할때사용하는연산자입니다.
예제 #1 struct student { int number; char name[10]; double grade; }; int main(void) { struct student s; s.number = 20070001; strcpy(s.name," 홍길동 "); s.grade = 4.3; 구조체선언 구조체변수선언 구조체멤버참조 printf(" 학번 : %d\n", s.number); printf(" 이름 : %s\n", s.name); printf( 학점 : %f\n", s.grade); return 0; } 학번 : 20070001 이름 : 홍길동학점 : 4.300000
struct student { int number; char name[10]; double grade; }; 구조체선언 예제 #2 학번을입력하시오 : 20070001 이름을입력하시오 : 홍길동학점을입력하시오 ( 실수 ): 4.3 학번 : 20070001 이름 : 홍길동학점 : 4.300000 int main(void) { struct student s; printf(" 학번을입력하시오 : "); scanf("%d", &s.number); 구조체변수선언 구조체멤버의주소전달 printf(" 이름을입력하시오 : "); scanf("%s", s.name); printf(" 학점을입력하시오 ( 실수 ): "); scanf("%lf", &s.grade); } printf(" 학번 : %d\n", s.number); printf(" 이름 : %s\n", s.name); printf(" 학점 : %f\n", s.grade); return 0;
#include <math.h> struct point { int x; int y; }; int main(void) { struct point p1, p2; int xdiff, ydiff; double dist; 예제 #3 점의좌표를입력하시오 (x y): 10 10 점의좌표를입력하시오 (x y): 20 20 두점사이의거리는 14.142136 입니다. p2 (x,y) printf(" 점의좌표를입력하시오 (x y): "); scanf("%d %d", &p1.x, &p1.y); p1 (x,y) } printf(" 점의좌표를입력하시오 (x y): "); scanf("%d %d", &p2.x, &p2.y); xdiff = p1.x - p2.x; ydiff = p1.y - p2.y; dist = sqrt(xdiff * xdiff + ydiff * ydiff); printf(" 두점사이의거리는 %f 입니다.\n", dist); return 0;
중간점검 1. 구조체안에선언된각각의변수들을 이라고한다. 2. 구조체의선언에사용하는키워드는 이다. 3. 구조체의태그는왜필요하며, 태그를사용하는경우과사용하지않은경우가어떻게다른가? 4. 구조체의선언만으로변수가만들어지는가? 5. 구조체의멤버를참조하는연산자는무엇인가?
구조체를멤버로가지는구조체 struct date { int year; int month; int day; }; // 구조체선언 struct student { // 구조체선언 int number; char name[10]; struct date dob; // 구조체안에구조체포함 double grade; }; struct student s1; // 구조체변수선언 s1.dob.year = 1983; s1.dob.month = 03; s1.dob.day = 29; // 멤버참조
#include <stdio.h> 예제 struct point { int x; int y; }; struct rect { struct point p1; struct point p2; }; int main(void) { struct rect r; int w, h, area, peri; p1(x,y) p2(x,y)
예제 printf(" 왼쪽상단의좌표를입력하시오 : "); scanf("%d %d", &r.p1.x, &r.p1.y); p1(x,y) printf(" 오른쪽상단의좌표를입력하시오 : "); scanf("%d %d", &r.p2.x, &r.p2.y); w = r.p2.x - r.p1.x; h = r.p2.x - r.p1.x; p2(x,y) area = w * h; peri = 2 * w + 2 * h; printf(" 면적은 %d 이고둘레는 %d 입니다.\n", area, peri); } return 0; 왼쪽상단의좌표를입력하시오 : 1 1 오른쪽상단의좌표를입력하시오 : 6 6 면적은 25 이고둘레는 20 입니다.
구조체변수의대입과비교 같은구조체변수까리대입은가능하지만비교는불가능하다. struct point { int x; int y; }; int main(void) { struct point p1 = {10, 20}; struct point p2 = {30, 40}; p2 = p1; // 대입가능 if( p1 == p2 ) // 비교 -> 컴파일오류!! printf("p1 와 p2 이같습니다.") } if( (p1.x == p2.x) && (p1.y == p2.y) ) printf("p1와 p2이같습니다.") // 올바른비교
중간점검 1. 구조체의변수끼리허용되는연산에는어떤것들이있는가? 2. 구조체태그와구조체변수의차이점은무엇인가? 3. 구조체멤버로구조체를넣을수있는가? 4. 구조체는배열을멤버로가질수있는가?
구조체배열 구조체를여러개모은것
구조체배열 구조체배열의선언 struct student { int number; char name[20]; double grade; }; int main(void) { struct student list[100]; // 구조체의배열선언 } list[2].number = 27; strcpy(list[2].name, " 홍길동 "); list[2].grade = 178.0;
구조체배열의초기화 구조체배열의초기화 struct student list[3] = { { 1, "Park", 172.8 }, { 2, "Kim", 179.2 }, { 3, "Lee", 180.3 } };
구조체배열예제 #define SIZE 3 struct student { int number; char name[20]; double grade; }; int main(void) { struct student list[size]; int i; for(i = 0; i < SIZE; i++) { printf(" 학번을입력하시오 : "); scanf("%d", &list[i].number); printf(" 이름을입력하시오 : "); scanf("%s", list[i].name); printf(" 학점을입력하시오 ( 실수 ): "); scanf("%lf", &list[i].grade); } 학번을입력하시오 : 20070001 이름을입력하시오 : 홍길동학점을입력하시오 ( 실수 ): 4.3 학번을입력하시오 : 20070002 이름을입력하시오 : 김유신학점을입력하시오 ( 실수 ): 3.92 학번을입력하시오 : 20070003 이름을입력하시오 : 이성계학점을입력하시오 ( 실수 ): 2.87 학번 : 20070001, 이름 : 홍길동, 학점 : 4.300000 학번 : 20070002, 이름 : 김유신, 학점 : 3.920000 학번 : 20070003, 이름 : 이성계, 학점 : 2.870000 } for(i = 0; i< SIZE; i++) printf(" 학번 : %d, 이름 : %s, 학점 : %f\n", list[i].number, list[i].name, list[i].grade); return 0;
중간점검 1. 상품 5 개의정보를저장할수있는구조체의배열을정의해보라. 상품은번호와이름, 가격을멤버로가진다.
구조체와포인터 구조체를가리키는포인터 포인터를멤버로가지는구조체 순서로살펴봅시다.
구조체를가리키는포인터 구조체를가리키는포인터 struct student *p; struct student s = { 20070001, " 홍길동 ", 4.3 }; struct student *p; p = &s; printf(" 학번 =%d 이름 =%s 학점 =%f \n", s.number, s.name, s.grade); printf(" 학번 =%d 이름 =%s 학점 =%f \n", (*p).number,(*p).name,(*p).grade);
-> 연산자 -> 연산자는구조체포인터로구조체멤버를참조할때사용 struct student *p; struct student s = { 20070001, " 홍길동 ", 180.2 }; struct student *p; p = &s; printf(" 학번 =%d 이름 =%s 키 =%f \n", s.number, s.name, s.grade); printf(" 학번 =%d 이름 =%s 키 =%f \n", (*p).number,(*p).name,(*p).grade); printf(" 학번 =%d 이름 =%s 키 =%f \n", p->number, p->name, p->grade);
-> 연산자 (*p).number p 가가리키는구조체변수 == p->number p 가가리키는구조체변수의멤버 number p 가가리키는구조체변수의멤버 number
예제 // 포인터를통한구조체참조 #include <stdio.h> struct student { int number; char name[20]; double grade; }; 학번 =20070001 이름 = 홍길동학점 =4.300000 학번 =20070001 이름 = 홍길동학점 =4.300000 학번 =20070001 이름 = 홍길동학점 =4.300000 int main(void) { struct student s = { 20070001, " 홍길동 ", 4.3}; struct student *p; p = &s; printf(" 학번 =%d 이름 =%s 키 =%f \n", s.number, s.name, s.grade); printf(" 학번 =%d 이름 =%s 키 =%f \n", (*p).number,(*p).name,(*p).grade); printf(" 학번 =%d 이름 =%s 키 =%f \n", p->number, p->name, p->grade); } return 0;
포인터를멤버로가지는구조체 struct date { int month; int day; int year; }; struct student { int number; char name[20]; double grade; struct date *dob; };
포인터를멤버로가지는구조체 int main(void) { struct date d = { 3, 20, 1980 }; struct student s = { 20070001, "Kim", 4.3 }; 학번 : 20070001 이름 : Kim 학점 : 4.300000 생년월일 : 1980 년 3 월 20 일 s.dob = &d; } printf(" 학번 : %d\n", s.number); printf(" 이름 : %s\n", s.name); printf(" 학점 : %f\n", s.grade); printf(" 생년월일 : %d년 %d월 %d일\n", s.dob->year, s.dob->month, s.dob->day ); return 0;
구조체와함수 구조체를함수의인수로전달하는경우 구조체의복사본이함수로전달되게된다. 만약구조체의크기가크면그만큼시간과메모리가소요된다. int equal(struct student s1, struct student s2) { if( strcmp(s1.name, s2.name) == 0 ) return 1; else return 0; }
구조체와함수 구조체의포인터를함수의인수로전달하는경우 시간과공간을절약할수있다. 원본훼손의가능성이있다. int equal(struct student const *p1, struct student const *p2) { } if( strcmp(p1->name, p2->name) == 0 ) else return 1; return 0; 포인터를통한구조체의변경을막는다.
구조체를반환하는경우 복사본이반환된다. struct student make_student(void) { struct student s; printf(" 나이 : ); scanf("%d", &s.age); printf(" 이름 : ); scanf("%s", s.name); printf(" 키 : ); scanf("%f", &s.grade); 구조체 s 의복사본이반환된다. } return s;
예제 #include <stdio.h> struct vector { float x; float y; }; struct vector get_vector_sum(struct vector a, struct vector b); int main(void) { struct vector a = { 2.0, 3.0 }; struct vector b = { 5.0, 6.0 }; struct vector sum; sum = get_vector_sum(a, b); printf(" 벡터의합은 (%f, %f) 입니다.\n", sum.x, sum.y); return 0; } a a + b b
예제 struct vector get_vector_sum(struct vector a, struct vector b) { struct vector result; result.x = a.x + b.x; result.y = a.y + b.y; a + b a b } return result; 벡터의합은 (7.000000, 9.000000) 입니다.
공용체 공용체 (union) 같은메모리영역을여러개의변수가공유 공용체를선언하고사용하는방법은구조체와아주비슷 union example { char c; int i; }; // 같은공간공유 // 같은공간공유
예제 #include <stdio.h> union example { int i; char c; }; int main(void) { union example v; 공용체선언 공용체변수선언. char 형으로참조. v.c = 'A'; printf("v.c:%c v.i:%i\n", v.c, v.i ); v.i = 10000; printf("v.c:%c v.i:%i\n", v.c, v.i); int 형으로참조. } v.c:a v.i:65 v.c:ꠕv.i:10000
ip 주소예제 #include <stdio.h> union ip_address { unsigned long laddr; unsigned char saddr[4]; }; int main(void) { union ip_address addr; addr.saddr[0] = 1; addr.saddr[1] = 0; addr.saddr[2] = 0; addr.saddr[3] = 127; printf("%x\n", addr.laddr); } return 0; 7f000001
중간점검 1. 공용체의선언에사용하는키워드는 이다. 2. 공용체에할당되는메모리의크기는어떻게결정되는가?
열거형 열거형 (enumeration) 이란변수가가질수있는값들을미리열거해놓은자료형 ( 예 ) 요일을저장하고있는변수는 { 일요일, 월요일, 화요일, 수요일, 목요일, 금요일, 토요일 } 중의하나의값만가질수있다.
열거형의선언 enum levels { low, medium, high }; 태그이름 값들을나열 int main(void) { enum levels english; // 열거형변수선언 english = high; // 변수에값대입 } 열거형변수선언
열거형초기화 enum levels1 { low, medium, high };// low=0, medium=1, high=2 enum levels2 { low=1, medium, high };// low=1, medium=2, high=3 enum levels3 { low=10, medium=20, high=30 }; 값을지정하기않으면 0 부터할당
열거형의예 enum days { SUN, MON, TUE, WED, THU, FRI, SAT }; enum colors { white, red, blue, green, black }; enum boolean { true, false }; enum months { JAN, FEB, MAR, APR, MAY, JUN, JUL, AUG, SEP, OCT, NOV, DEC }; enum major { COMMUNICATION, COMPUTER, ELECTRIC, ELECTRONICS }; enum component { MAIN_BOARD, CPU, GRAPHIC_CARD, DISK, MEMORY }; enum levels { low = 1, medium, high }; enum CarOptions { SunRoof = 0x01, Spoiler = 0x02, FogLights = 0x04, TintedWindows = 0x08, }
열거형과다른방법과의비교 정수사용기호상수열거형 switch(code) { case 1: printf("lcd TV\n"); break; case 2: printf("pdp TV\n"); break; } 컴퓨터는알기쉬우나사람은기억하기어렵다. #define LCD 1 #define PDP 2 switch(code) { case LCD: printf("lcd TV\n"); break; case PDP: printf("pdp TV\n"); break; } 기호상수를작성할때오류를저지를수있다. enum tvtype { LCD, PDP }; enum tvtype code; switch(code) { case LCD: printf("lcd TV\n"); break; case PDP: printf("pdp TV\n"); break; } 컴파일러가중복이일어나지않도록체크한다.
예제 // 열거형 #include <stdio.h> enum days { MON, TUE, WED, THU, FRI, SAT, SUN }; char *days_name[] = { "monday", "tuesday", "wednesday", "thursday", "friday", "saturday", "sunday" }; int main(void) { enum days d; } for(d=mon; d<=sun; d++) { printf("%d 번째요일의이름은 %s 입니다 \n", d, days_name[d]); } 0번째요일의이름은 monday입니다 1번째요일의이름은 tuesday입니다 2번째요일의이름은 wednesday입니다 3번째요일의이름은 thursday입니다 4번째요일의이름은 friday입니다 5번째요일의이름은 saturday입니다 6번째요일의이름은 sunday입니다
enum tvtype { tube, lcd, plasma, projection }; 예제 int main(void) { enum tvtype type; printf("tv 종류코드를입력하시오 : "); scanf("%d", &type); switch(type) { case tube: printf(" 브라운관 TV를선택하셨습니다.\n"); break; case lcd: printf("lcd TV를선택하셨습니다.\n"); break; case plasma: printf("pdp TV를선택하셨습니다.\n"); break; case projection: printf(" 프로젝션 TV를선택하셨습니다.\n"); break; default: printf(" 다시선택하여주십시오.\n"); break; } return 0; } TV 종류코드를입력하시오 : 3 프로젝션 TV 를선택하셨습니다.
중간점검 1. 열거형의선언에사용하는키워드는 이다. 2. 열거형은어떤경우에사용되는가? 3. 열거형에서특별히값을지정하지않으면자동으로정수상수값이할당되는가?
typedef 의개념
typedef typedef 은새로운자료형 (type) 을정의 (define) C 의기본자료형을확장시키는역할 typedef old_type new_type; 새로운자료형을정의 typedef unsigned char BYTE; 기존의자료형 새로운자료형
typedef 의예 typedef unsiged char BYTE; BYTE index; // unsigned int index; 와같다. typedef int INT32; typedef unsigned int UINT32; INT32 i; // int i; 와같다. UINT32 k; // unsigned int k; 와같다.
구조체로새로운타입정의 구조체로새로운타입을정의할수있다. struct point { int x; int y; }; typedef struct point POINT; POINT a, b;
#include <stdio.h> 예제 typedef struct point { int x; int y; } POINT; POINT translate(point p, POINT delta); int main(void) { POINT p = { 2, 3 }; POINT delta = { 10, 10 }; POINT result; result = translate(p, delta); printf(" 새로운점의좌표는 (%d, %d) 입니다.\n", result.x, result.y); } return 0;
예제 POINT translate(point p, POINT delta) { POINT new_p; new_p.x = p.x + delta.x; new_p.y = p.y + delta.y; } return new_p; 새로운점의좌표는 (12, 13) 입니다.
typedef 과 #define 비교 이식성을높여준다. 코드를컴퓨터하드웨어에독립적으로만들수있다 ( 예 ) int 형은 2 바이트이기도하고 4 바이트, int 형대신에 typedef 을이용한 INT32 나 INT16 을사용하게되면확실하게 2 바이트인지 4 바이트인지를지정할수있다. #define 을이용해도 typedef 과비슷한효과를낼수있다. 즉다음과같이 INT32 를정의할수있다. #define UINT32 unsigned int typedef float VECTOR[2];// #define 으로는불가능하다. 문서화의역할도한다. typedef 을사용하게되면주석을붙이는것과같은효과
중간점검 1. typedef 의용도는무엇인가? 2. typedef 의장점은무엇인가? 3. 사원을나타내는구조체를정의하고이것을 typedef 을사용하여서 employee 라는새로운타입으로정의하여보자.
Q & A