'프로그래밍 > C언어' 카테고리의 다른 글
함수 포인터 (0) | 2015.11.24 |
---|---|
[visual studio 2013] scanf_s에서 s없애기와 freopen (0) | 2015.10.24 |
Visual Studio 설치시 설정 및 Hello world! (0) | 2015.09.11 |
구조체 포인터의 선언과 접근( -> ) (0) | 2015.09.07 |
구조체 선언에 typedef 사용하기 (0) | 2015.09.07 |
함수 포인터 (0) | 2015.11.24 |
---|---|
[visual studio 2013] scanf_s에서 s없애기와 freopen (0) | 2015.10.24 |
Visual Studio 설치시 설정 및 Hello world! (0) | 2015.09.11 |
구조체 포인터의 선언과 접근( -> ) (0) | 2015.09.07 |
구조체 선언에 typedef 사용하기 (0) | 2015.09.07 |
-함수 포인터(function pointer)
함수의 메모리 주소를 가지는 포인터 변수
-함수 포인터의 선언
만약 함수 포인터 이름에다가 소괄호를 붙이지 않고 다음과 같이 선언하면 무엇을 의미할까?
int* pFunc(int, int)
위 문장은 반환형이 int형 포인터인 함수를 선언한 것이다.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 | #include <stdio.h> int add(int x, int y) { return x + y; } int main(void) { int sum; int(*pFunc)(int, int) = NULL; pFunc = add; sum = pFunc(16, 62); //sum = (*pFunc)(16, 62); //sum = add(16, 62); //sum = (*add)(16, 62); printf("16 + 62 = %d\n", sum); return 0; } | cs |
-함수 포인터 자료형
함수 포인터를 선언할 때 특정 반환값과 인자를 가지는 함수 포인터를 반복적으로 만들기 귀찮기 때문에, 이것을 자료형으로 아예 만들 수 있다.
-함수 포인터 자료형의 선언
typedef void (*funcName_1)(int);
typedef void funcName_2(int);
둘 다 함수 포인터의 자료형이지만 둘의 차이가 있다.
만약 void temp(int)의 주소값을 전달 받는 다면
funcName_1 fc = temp
는 옳은 문장이지만 funcName_2 fc = temp는 컴파일 에러가 난다. 이를 해결하기 위해서는
funcName_2 * fc = temp;
와 같이 사용해야 한다. 다만 매개변수로 함수의 주소값을 전달 받는 경우에는 *의 생략이 가능하다.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 | #include <stdio.h> #include <stdlib.h> typedef int BTData; typedef struct _bTreeNode { BTData data; struct _bTreeNode* left; struct _bTreeNode* right; }BTreeNode; BTreeNode* MakeBTreeNode(void) { BTreeNode* nd = (BTreeNode*)malloc(sizeof(BTreeNode)); nd->left = NULL; nd->right = NULL; return nd; } BTData GetData(BTreeNode* bt) { return bt->data; } void SetData(BTreeNode* bt, BTData data) { bt->data = data; } BTreeNode* GetLeftSubTree(BTreeNode* bt) { return bt->left; } BTreeNode* GetRightSubTree(BTreeNode* bt) { return bt->right; } void MakeLeftSubTree(BTreeNode* main, BTreeNode* sub) { if (main->left != NULL) free(main->left); main->left = sub; } void MakeRightSubTree(BTreeNode* main, BTreeNode* sub) { if (main->right != NULL) free(main->right); main->right = sub; } typedef void VisitFuncPtr(BTData data); void PreorderTraverse(BTreeNode* bt, VisitFuncPtr action) { if (bt == NULL) return; action(bt->data); PreorderTraverse(bt->left, action); PreorderTraverse(bt->right, action); } void ShowIntData(int data) { printf("%d ", data); } int main(void) { BTreeNode* bt1 = MakeBTreeNode(); BTreeNode* bt2 = MakeBTreeNode(); BTreeNode* bt3 = MakeBTreeNode(); BTreeNode* bt4 = MakeBTreeNode(); SetData(bt1, 1); SetData(bt2, 2); SetData(bt3, 3); SetData(bt4, 4); MakeLeftSubTree(bt1, bt2); MakeRightSubTree(bt1, bt3); MakeLeftSubTree(bt2, bt4); printf("Preorder Traverse: \n"); PreorderTraverse(bt1, ShowIntData); printf("\n"); return 0; } | cs |
sizeof (0) | 2016.05.22 |
---|---|
[visual studio 2013] scanf_s에서 s없애기와 freopen (0) | 2015.10.24 |
Visual Studio 설치시 설정 및 Hello world! (0) | 2015.09.11 |
구조체 포인터의 선언과 접근( -> ) (0) | 2015.09.07 |
구조체 선언에 typedef 사용하기 (0) | 2015.09.07 |
scanf_s에서 s없애기
프로그램 맨 위에
#pragma warning(disable:4996)
를 붙여주면 visual studio 2013에서 scanf_s나 freopen_s 처럼 _s 를 붙여줘야하는 것을 피할 수 있다.
freopen
freopen은 프로그래밍 문제 풀때 입력을 자동으로 받아주는 함수이다.
프로그램의 프로젝트안에 text파일을 넣어주고 함수를 쓰면 text파일 안에있는 값을 자동으로 받아서 키보드로 입력한것처럼 동작하게 한다.
text파일이 다음과 같이 저장해놓고 프로그램에서 freopen을 쓴다.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26 |
#include <stdio.h>
#pragma warning(disable:4996)
int main(void)
{
int d[20];
if (freopen("sample_input.txt", "r", stdin) == NULL)
fprintf(stderr, "error redirecting stdin\n");
for (int i = 0; i < 20; i++){
scanf("%d", &d[i]);
}
for (int i = 0; i < 20; i++){
printf("%d ", d[i]);
}
printf("\n");
return 0;
} |
cs |
sizeof (0) | 2016.05.22 |
---|---|
함수 포인터 (0) | 2015.11.24 |
Visual Studio 설치시 설정 및 Hello world! (0) | 2015.09.11 |
구조체 포인터의 선언과 접근( -> ) (0) | 2015.09.07 |
구조체 선언에 typedef 사용하기 (0) | 2015.09.07 |
visual studio 설치 설정
Hello world
함수 포인터 (0) | 2015.11.24 |
---|---|
[visual studio 2013] scanf_s에서 s없애기와 freopen (0) | 2015.10.24 |
구조체 포인터의 선언과 접근( -> ) (0) | 2015.09.07 |
구조체 선언에 typedef 사용하기 (0) | 2015.09.07 |
[흐름 변화] - if, switch, while, for, do while, continue, break (0) | 2015.09.06 |
구조체 포인터라고 해서 다를 것 없다.
int형 포인터 변수를 선언하듯이 선언하면 되고, int형 포인터를 통해서 접근하듯이 접근하면 된다.
-구조체 포인터
함수의 인자로 기본자료형 변수와 마찬가지로 구조체 변수도 포인터로 전달 할 수 있다.
이렇게 함수 내부에서 외부에 있는 구조체 변수에 직접 접근하기 위해 구조체 포인터를 쓴다.
구조체 변수의 포인터 선언 방식과 매개변수로 구조체 포인터를 선언하는 방식은 일반적은 변수의 포인터 선언규칙을 그대로 적용하면 된다.
또한 구조체 변수의 주소 값을 얻을 때고 & 연산자를 사용한다.
-포인터를 이용한 구조체 멤버의 접근
(*구조체_포인터).멤버
여기에서 (*구조체_포인터)는 구조체 변수를 의미하게 된다.
멤버에 접근하는 데 ( )가 필요한 이유는 .연산자보다 *연산자의 우선순위가 낮기 때문이다.
그런데 포인터를 이용한 구조체 멤버의 접근을 간단히 처리하는 -> 연산자가 있는 데 다음과 같은 방식으로 접근이 가능하다.
구조체_포인터->멤버
즉, "(*구조체_포인터).멤버"는 "구조체_포인터->멤버"와 완전히 동일한 연산문이다.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 | #include <stdio.h> #include <string.h> #define NAME_LEN 30 #define PID_LEN 15 typedef struct _person { char name[NAME_LEN]; //이름 char ID[PID_LEN]; //주민등록 번호 unsigned int age; //나이 }person; void ShowPersonData(person * ptr) { printf("이름: %s \n", (*ptr).name); printf("주민등록번호: %s \n", (*ptr).ID); printf("나이: %u \n", ptr->age); } int main(void) { person A_youjin = { "안유진", "920720-2587412", 24 }; person * personPtr = &A_youjin; ShowPersonData(personPtr); while (1); } | cs |
[visual studio 2013] scanf_s에서 s없애기와 freopen (0) | 2015.10.24 |
---|---|
Visual Studio 설치시 설정 및 Hello world! (0) | 2015.09.11 |
구조체 선언에 typedef 사용하기 (0) | 2015.09.07 |
[흐름 변화] - if, switch, while, for, do while, continue, break (0) | 2015.09.06 |
구조체의 정의 (0) | 2015.09.04 |
1.구조체를 선언할 때 typedef 사용
구조체 변수를 선언 할 때에는 다음과 같이 한다.
struct 자료형_이름 변수_이름;
일반적인 자료형인 int나 double과 마찬가지로 [자료형_이름 변수_이름;] 이렇게 선언하고 싶을 것이다.
앞에있는 struct를 떼버리고...
그런데 typedef 선언을 이용하면 이것이 실제로 가능하다.
typedef struct 자료형_이름 새롭게_축약된_자료형_이름;
이 선언에 의해서 새롭게_축약된_자료형_이름 이 struct 자료형_이름 을 대신 할 수 있는 것이다.
이 후 변수를 선언할 때, 컴파일러에 의하여 struct 자료형_이름과 새롭게_축약된_자료형_이름이 둘 다 사용될 수 있다.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 | #include <stdio.h> struct point { double xpos; double ypos; }; typedef struct point point; int main(void) { point p1; struct point p2; p1.xpos = 0.1, p1.ypos = 0.2; p2.xpos = 2.4, p2.ypos = 2.5; printf("X축 거리: %g \n", p2.xpos - p1.xpos); printf("Y축 거리: %g \n", p2.ypos - p1.ypos); while (1); } | cs |
2.구조체의 정의에 포험되는 typedef
위의 코드에서는 구조체의 정의와 typedef선언을 별도로 하고 있다.
하지만 이 둘을 다음과 같이 묶어서 선언할 수 있다.
struct point { double xpos; double ypos; }; typedef struct point point; |
typedef struct point { double xpos; double ypos; } point; |
아래의 코드는 위의 코드를 오른편에 선언방식으로 대체한 것이다.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 | #include <stdio.h> typedef struct point { double xpos; double ypos; }point; int main(void) { point p1; struct point p2; p1.xpos = 0.1, p1.ypos = 0.2; p2.xpos = 2.4, p2.ypos = 2.5; printf("X축 거리: %g \n", p2.xpos - p1.xpos); printf("Y축 거리: %g \n", p2.ypos - p1.ypos); while (1); } | cs |
Visual Studio 설치시 설정 및 Hello world! (0) | 2015.09.11 |
---|---|
구조체 포인터의 선언과 접근( -> ) (0) | 2015.09.07 |
[흐름 변화] - if, switch, while, for, do while, continue, break (0) | 2015.09.06 |
구조체의 정의 (0) | 2015.09.04 |
조건부 컴파일을 위한 매크로 (0) | 2015.09.04 |
Visual Studio 설치시 설정 및 Hello world! (0) | 2015.09.11 |
---|---|
구조체 포인터의 선언과 접근( -> ) (0) | 2015.09.07 |
구조체 선언에 typedef 사용하기 (0) | 2015.09.07 |
구조체의 정의 (0) | 2015.09.04 |
조건부 컴파일을 위한 매크로 (0) | 2015.09.04 |
구조체를 사용함으로써 여러 변수를 하나로 묶어 새로운 자료형을 정의 할 수 있다.
-구조체 정의
struct 자료형_이름
{
변수 선언1;
변수 선언2;
변수 선언3;
};
배열도 변수이기 때문에 구조체의 멤버가 될 수 있다.
-구조체 변수 선언
struct 자료형_이름 변수_이름;
또는 다음과 같이 구조체를 정의함과 동시에 구조체 변수를 선언 할 수 있다.
struct 자료형_이름
{
변수 선언1;
변수 선언2;
변수 선언3;
}변수_이름1, 변수_이름2, 변수_이름3;
-구조체 변수의 초기화
구조체 변수의 초기화 방법은 배열의 초기화와 유사하다.
단순히 멤버의 순서대로 초기화할 대상을 나열하면 된다.
그래서 초기화 과정에서는 문자열 저장을 위해서 strcpy()함수를 호출 하지 않아도 된다.
-구조체 변수의 멤버에 접근
구조체_변수_이름.멤버_이름
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 | #include <stdio.h> #include <string.h> struct person //구조체 정의 { char name[20]; char phoneNumber[20]; int age; }man1, man2; //구조체 변수 선언 int main(void) { //struct person man1, man2; //구조체 변수 선언 /*구조체 변수의 초기화*/ struct person man3 = {"홍길동", "010-1234-8697", 84}; /*구조체 멤버에 접근*/ strcpy(man1.name,"안유진"); strcpy(man1.phoneNumber,"010-4085-8697"); man1.age = 23; /*구조체의 scanf 사용*/ printf("이름을 입력하세요: "); scanf("%s", man2.name); printf("전화 번호를 입력하세요: "); scanf("%s", man2.phoneNumber); printf("나이를 입력하세요: "); scanf("%d", &man2.age); /*구조체의 printf 사용*/ printf("이름: %s \n", man1.name); printf("전화번호: %s \n", man1.phoneNumber); printf("나이: %d\n\n", man1.age); printf("이름: %s \n", man2.name); printf("전화번호: %s \n", man2.phoneNumber); printf("나이: %d\n\n", man2.age); printf("이름: %s \n", man3.name); printf("전화번호: %s \n", man3.phoneNumber); printf("나이: %d\n", man3.age); return 0; } |
Visual Studio 설치시 설정 및 Hello world! (0) | 2015.09.11 |
---|---|
구조체 포인터의 선언과 접근( -> ) (0) | 2015.09.07 |
구조체 선언에 typedef 사용하기 (0) | 2015.09.07 |
[흐름 변화] - if, switch, while, for, do while, continue, break (0) | 2015.09.06 |
조건부 컴파일을 위한 매크로 (0) | 2015.09.04 |
컴파일 할 때, 특정 조건에 따라 소스코드의 일부를 삽입할 수 있는 지시자들이 있다.
이 지시자에서 조건을 만족하지 않아서 삽입되지 않은 코드는 오류가 있더라도 전체 프로그램은 컴파일 오류가 생기지 않는다.
1. #if... #endif : 매크로가 참이면
#if (매크로)
코드
#endif
매크로가 참이라면 코드가 소스에 삽입되고, 거짓이면 코드는 삭제된다.
매크로 부분에 연산자는 사용가능하나 변수는 사용할 수 없다.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 | #include <stdio.h> #define ADD #define MIN int main(void) { int num1, num2; //int add = 0, min = 1; printf("두 개의 정수 입력: "); scanf_s("%d %d", &num1, &num2); #if ADD printf("%d + %d = %d\n", num1, num2, num1 + num2); #endif // ADD #if MIN printf("%d - %d = %d\n", num1, num2, num1 - num2); #endif // MIN return 0; } |
2. #ifdef... #endif : 매크로가 정의되어 있다면
#ifdef (매크로)
코드
#endif
매크로가 #define으로 정의가 되어 있다면 코드가 삽입된다.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 | #include <stdio.h> #define ADD 0 #define MIN 1 int main(void) { int num1, num2; //int add = 0, min = 1; printf("두 개의 정수 입력: "); scanf_s("%d %d", &num1, &num2); #ifdef ADD printf("%d + %d = %d\n", num1, num2, num1 + num2); #endif // ADD #ifdef MIN printf("%d - %d = %d\n", num1, num2, num1 - num2); #endif // MIN while (1); } | cs |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 | #include <stdio.h> #define HIT_NUM 6 int main(void) { #if HIT_NUM == 5 puts("1"); #else puts("2"); #endif while (1); } | cs |
#else + if인 것으로 조건을 추가할 수 있다.
#if의 경우에만 사용할 수 있다.
매크로1이 참이면 코드1을 수행, 매크로1이 거짓이고 매크로2가 참이면 코드2를 수행한다.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 | #include <stdio.h> #define HIT_NUM 6 int main(void) { #if HIT_NUM == 5 puts("5"); #elif HIT_NUM == 6 puts("6"); #else puts("nope"); #endif while (1); } | cs |
Visual Studio 설치시 설정 및 Hello world! (0) | 2015.09.11 |
---|---|
구조체 포인터의 선언과 접근( -> ) (0) | 2015.09.07 |
구조체 선언에 typedef 사용하기 (0) | 2015.09.07 |
[흐름 변화] - if, switch, while, for, do while, continue, break (0) | 2015.09.06 |
구조체의 정의 (0) | 2015.09.04 |