기본
// 선언
struct date {
	int year;
    int month;
    int day;
};


// 사용
struct date date;

date.year = 2020;
date.month = 1;
date.day = 1;

// typedef 선언
typedef struct {
	int year;
	int month;
	int day;
} date_t;

// 사용
date_t date;

날짜는 년,월,일을 모두 표현해야 의미가 있다. 날짜의 year, month, day처럼 의미있는 단위들을 모아서 선언 및 나열 하는 것이 바로 구조체다. (다른언어의 Class와 비슷하다. 하지만 구조체는 함수 즉 메소드를 가지지 않는다.)

 

사용은 struct date 변수명과 같이 사용 하며 구조체의 접근은 . 으로 접근한다. struct date 변수명이 너무 지저분하게 느껴질때는 typedef와 함께 선언해서 사용하면 date_t date와 같이 깔끔하게 사용할 수 있다.

 

 

기계의 입장에서

C언어 포큐아카데미

왼쪽은 단순히 나열, 오른쪽은 구조체를 사용해서 하나로 묶었다. 이때 기계는 구조체를 어떻게 인식할까?

C언어 포큐아카데미

결론은 똑같다. 컴퓨터 입장에서 구조체가 뭔지 모른다 그냥 사람이 편하게 코딩 할 수 있게 묶는 용도일 뿐이다. 이래서 C언어와 같은 언어를 사람이 이해하기 쉬운 하이레벨랭귀지라고 부른단다.

 

 

구조체 매개변수

C언어 포큐아카데미

date라는 구조체를 매개변수로 삼는 do_something_useless라는 함수가  있다. 구조체 매개변수는 값복사 임으로 원본의 구조체를 변경하지 못한다. 매개변수로 구조체를 던질때 그냥 일반 자료형이랑 똑같이 작동한다는 말이다.

 

그렇다면 원본을 변경하기 위해서는 어떻게 해야할까?? 구조체 포인터를 사용 하면된다.

 

구조체 포인터
typedef struct {
	int year;
	int month;
	int day;
} date_t;

void increate_year(date_t* date) 
{
	(*date).year = (*date).year + 1;
}

void main()
{

	date_t date;

	date.year = 2020;
	printf("%d\n", date.year);
	increate_year(&date);
	printf("%d\n", date.year);

}

C언어 포큐아카데미

위와 같이 구조체 주소를 던지고 함수 내부에서는 포인터로 원본에 값을 접근해서 +1 을한다.

 

C언어 포큐아카데미

(*date).이 조금 지저분할때는 date->를 사용하면 좀 더 편하게 구조체를 접근할 수 있다.

 

 

값복사 vs 주소만 넘기기

 

C언어 포큐아카데미

위와 같이 int형 5만개의 구조체를 매개변수로 넘긴다면 값복사가 5만개나 되서 50,000 X 4 = 200KB의 스택 메모리를 사용하는 엄청난 일이 벌어진다. 이럴땐 구조체의 주소를 넘겨서 포인터로 간단하게 마무리하자.

'C' 카테고리의 다른 글

C 언어 변수에 함수 할당  (0) 2019.12.20
C언어 얕은복사 vs 깊은복사  (0) 2019.12.18
C언어 포인터 배열  (0) 2019.12.13
C 포인터 const  (0) 2019.12.12
C언어 배열과 포인터 차이점  (0) 2019.12.12

포인터 배열이란 무엇인가..... 말 그대로 포인터 변수를 저장하는 배열이다.

int nums[3]은 int를 저장하는 배열 int* nums[3] int*를 저장하는 배열이다.

 

 

C언어 포큐아카데미

num_pointers라는 포인터 배열은 nums1, nums2, num3의 주소를 저장하는 배열이다. 

num_pointers[0][0], num_pointers[0][1] 과 같이 2차원 배열처럼 접근 할 수있다.

 

그럼 2차원 배열이랑 포인터 배열이랑 똑같네?? 아니!!!!

 

2차원 배열과 포인터 배열이 같다?(아니다)

에러(C언어 포큐아카데미)

 

matrix라는 2차원 배열을 만들고 do_magic에서는 포인터 배열로 받았다 왜냐? 둘은 똑같으니깐

이렇게 하면 컴파일 에러가 난다!. 왜냐? 2차원 배열과 포인터 배열은 다르니깐

 

2차원 배열은 한덩어리 메모리라 주솟값이 저장된 곳이 없다. 반면 포인터 배열은 각각의 요소에 주소가 저장된다.

(밑에서 더 자세히)

 

 

수정(C언어 포큐아카데미)

위와 같이 2차원 배열은 2차원 배열로 받아야한다. 그래야 컴파일러가 offset을 이해할 수 있다.

 

 

 

#include <stdio.h>


int main(void)
{
    const int nums[3][5] = {
        { 35, 50, 65, 24, 71 }, 
        { 52, 93, 30, 18, 83 },
        { 72, 53, 41, 88, 69 }
    };
    const int* nums2[3];
    
    nums2[0] = nums[0];
    nums2[1] = nums[1];
    nums2[2] = nums[2];
    
    printf("nums[0] address: %p\n", (void*)nums[0]);
    printf("nums[1] address: %p\n", (void*)nums[1]);
    printf("nums[2] address: %p\n", (void*)nums[2]);
    printf("nums[2]'s offset from nums[0]: %d\n", nums[2] - nums[0]);         /* 10 */
    printf("nums[1]'s offset from nums[0]: %d\n", nums[1] - nums[0]);         /* 5 */
    
    printf("\n");
    
    printf("nums2[0] address: %p\n", (void*)&nums2[0]);
    printf("nums2[1] address: %p\n", (void*)&nums2[1]);
    printf("nums2[2] address: %p\n", (void*)&nums2[2]);
    printf("nums2[2]'s offset from nums2[0]: %d\n", &nums2[2] - &nums2[0]);   /* 2 */
    printf("nums2[1]'s offset from nums2[0]: %d\n", &nums2[1] - &nums2[0]);   /* 1 */
    
    return 0;
}

nums는 2차원배열, nums2는 포인터 배열이다. 주목해야 할 점은 offset(얼마나 떨어져있냐)의 값이다.

2차원 배열의 경우 nums[0]과 nums[1]은 5만큼 떨어져있다. 그 말을 풀어서 이야기하면 nums[0]인 35와 nums[1]인 52 사이에 5개의 메모리 블럭이 존재한다고 보면 되겠다.(좀 더 정확히 하면 4바이트 * 5 = 20바이트 만큼 떨어짐)

 

반면 포인터 배열의 경우 nums2[0]과 nums[1]의 offset은 1이다. 왜냐하면 각각의 메모리 블럭에 주소가 저장되어 있기 때문이다.(하나의 포인터의 크기만큼 떨어져있다)

 

- C언어 포큐 아카데미 복습 -

'C' 카테고리의 다른 글

C언어 얕은복사 vs 깊은복사  (0) 2019.12.18
C언어 구조체  (0) 2019.12.18
C 포인터 const  (0) 2019.12.12
C언어 배열과 포인터 차이점  (0) 2019.12.12
C 스택 메모리 해부(어셈블리어)  (3) 2019.12.09

const 변수를 불변하게 만들고 싶을때 사용한다.(immutable)

요즘 js, React쪽에서는 웬만한 변수는 다 const로 바르고 그게 아닌 경우에만 let(mutable)로 선언한다.

 

C에서도 const를 당연히 사용하는데 포인터에 사용할 때 귀여운 상황이 벌어진다.

 

1. 포인터가 담은 값(주소)를 불변하게 할 것인가

2. 포인터가 담은 값(주소)로 간뒤에 그 값을 불변하게 할 것인가(* 로 접근)

 

 

주소냐, 값이냐

C언어 포큐아카데미

왼쪽의 int* const p = &num은 포인터의 값 즉 주소를 변경하는 것을 막는 코드이다.(1번)

오른쪽의 const int* p = &num은 포인터가 가르키는 값을 변경하는 것을 막는 코드이다.(2번)

실제로 2번 값을 막는 것이 훨씬 많이 쓰이며 중요하다고 한다.

 

'C' 카테고리의 다른 글

C언어 구조체  (0) 2019.12.18
C언어 포인터 배열  (0) 2019.12.13
C언어 배열과 포인터 차이점  (0) 2019.12.12
C 스택 메모리 해부(어셈블리어)  (3) 2019.12.09
C언어 Char, int 정리  (0) 2019.12.05

배열은 주소다. 포인터 변수도 주소를 값으로 가진다.

 

배열과 포인터는 맥락상 거의 같다. 하지만 포인터에서는 먹히고 배열에서는 안되는게 몇가지가 있다.

 

이 차이점을 분명히 하면서 포인터에 대한 이해도를 높이자.(포큐아카데미 설명이 정말 자세하다)

 

sizeof

C언어 포큐아카데미

sizeof으로 크기를 알고 싶을때 포인터는 변수 임으로 4바이트(사용하는 bit수에 따라 다름)를 반환값으로 가지지만 배열은 해당 원소의 개수 X 자료형의 크기를 반환한다.

 

 

배열 문자열

C언어 포큐아카데미

배열로 Monday 라는 문자열을 만들게 되면 day1 변수는 스택 메모리에 저장되며 값 또한 실제로 존재한다.

day[0]은 "M" day[1]은 "o" ... 이런식으로 물론 변경도 가능!

 

 

포인터 문자열

C언어 포큐아카데미

포인터로 day2 문자열 변수를 만들게 되면 day2라는 변수는 스택에 저장 되지만 값으로는 "Monday" 문자열의 주소값을 가짐 또한 "Monday" 문자열의 메모리 저장 공간은 데이터 영역임.(값 변경 불가능 ReadOnly)

 

 

포인터, 배열 대입 차이점

배열로 만든 문자열 수정 가능 BUT 포인터로 만든 문자열 수정 불가

 

배열은 상수다! 이말을 기억하면 안되는것들을 판단할때 참 편하다. 위 처럼 포인터변수는 값을 대입할 수 있지만 배열 상수는 불가!

 

 

포인터 변수는 증감을 할 수 있지만 배열 상수는 불가!

 

 

'C' 카테고리의 다른 글

C언어 구조체  (0) 2019.12.18
C언어 포인터 배열  (0) 2019.12.13
C 포인터 const  (0) 2019.12.12
C 스택 메모리 해부(어셈블리어)  (3) 2019.12.09
C언어 Char, int 정리  (0) 2019.12.05
스택메모리

 

각 함수에서 사용하는 지역 변수, 매개 변수 등을 임시적으로 저장하는 공간이다.

스택 메모리의 크기는 프로그램 빌드 시에 결정되며 위치는 실행 시에 결정된다.

 

간단한 예제(포큐 아카데미 C 강좌)

 

위의 간단한 예제로 설명하자면 main 함수의 int a, int b 그리고 add 함수로 넘겨주는 매개변수 a, b 모두 지역, 매개변수 임으로 스택메모리에 값이 저장된다.

 

C언어에서는 New라는 키워드가 없는데 이는 기본적으로 변수를 선언하면 변수가 스택메모리에 할당되기 때문이다.

C언어 기본 자료형 변수(char, int, float)는 스택 메모리에 할당되며 기본 자료형을 함수 매개변수로 전달하면 스택에 복사본을 만든다(값복사, 값형) 

반대로 다른언어에서 처럼 malloc을 쓰면 Heap 영역에 할당된다. Heap 영역에 메모리를 할당하면 운영체제가 프로그래머가 요구한 메모리 사이즈를 검색하고 반환하며 크기가 가변적이라 메모리에 구멍이 숭숭뚫린다.

 

 

스택 메모리 조금 더 자세히

스택 메모리(포큐 아카데미 C 강좌)

 

1. 스택은 큰주소에서 작은 주소로 쌓인다

2. EBP(Extended Base Pointer) : 현재 스택 프레임의 첫 주소(Base 주소)

3. ESP(Extened Stack Pointer) : 현재 스택 포인터

4. 스택 프레임(stack frame) : 각 함수가 사용하는 스택 메모리의 범위

 

스택 메모리 + 어셈블리어

 

어셈블리어 Mov(포큐 아카데미 C 강좌)

왼쪽은 어셈블리어 오른쪽은 그에 대응되는 스택 메모리의 그림이다.

1. esp를 ebp로 Move해라! 그러니깐 같은 것을 가르켜라

 

 

어셈블리어 메모리 영역 확장(포큐 아카데미 C 강좌)

2. esp를 14h(16진수) 즉 십진수 20만큼 sub(빼라) -> ESP 영역을 20만큼 확장해라 한 블럭이 4바이트니 5칸 밑으로 ESP를 내려라

 

어셈블리어 값 할당 (포큐 아카데미 C 강좌)

3. 주소 ebp - 4 즉 0x0073FE68에 0을 넣어라. ebp - 8 주소에 1을 넣어라, ebp -0Ch 주소에 2를 넣어라

 

어셈블리어 매개변수 전달 (포큐 아카데미 C 강좌)

4-1. eax에 ebp-0Ch의 값을 넣고, ecx에 ebp-8의 값을 넣어라(eax, ecx는 CPU 레지스터 영역임) 

4-2. 아까 담은 ecx의 값을 현재 스택 포인터인 esp에 넣어라 그리고 esp+4에는 eax를 넣어라

 

방금 4줄의 어셈블리어가 add 함수에 매개변수를 전달하는 방식이다

 

add 함수 Call 및 돌아갈 주소 push (포큐 아카데미 C 강좌)

5. 00011040이라는 주소를 Call한다(add 함수) 그리고 다시 돌아올 주소 0001102D를 ESP-14h 주소에 add한다.

 

add 함수 시작 전(포큐 아카데미 C 강좌)

6. 호출자 함수의 EBP를 PUSH 한다. 호출된 add 함수가 끝나고 다시 main함수로 돌아가야하니깐

 

add 함수 시작 전(포큐 아카데미 C 강좌)

7. EBP를 ESP로 move한다

 

add 함수 스택 메모리 확보 (포큐 아카데미 C 강좌)

 

8. ESP를 0c(12) 만큼 늘림으로서 add 함수의 스택 메모리 영역 확보

 

add 함수 res 변수 생성(포큐 아카데미 C 강좌)

 

9-1. eax, ecx, edx 에 2,1,1을 담고 dex에 ebp+0Ch 값을 더한다 1+2 = 3 

9-2. edx의 값을 ebp-4에 할당한다 즉(res 변수)

 

 

add 함수 인자 생성 (포큐 아카데미 C 강좌)

 

10-1. edx에 ebp-4를 할당(의미없음)

10-2. eax의 값(2)를 ebp -8에 할당

10-3, edx의 값을 eax에 할당(의미없음)

10-4, ecx의 값(1)을 ebp-0c에 할당 

 

add 함수 스택 영역 해제(포큐 아카데미 C 강좌)

11. esp에 0c를 add 함으로서 add 함수의 스택 영역을 해제한다. 여기서 중요한점은 3,2,1과 같은 값을 지우지 않는다는 점이다. 나중에 다른 함수가 해당 주소를 다시  읽었을때 3,2,1 같은 쓰레기 데이터가 남아있는 이유가 이때문이다.

 

add 함수 스택 영역 해제(포큐 아카데미 C 강좌)

12. 0x0073FE6C의 주소로 ebp를 이동하면서 pop

 

add 함수 종료 return (포큐 아카데미 C 강좌)

13. return 명령어 실행 : 0x0001102D 의 주소로 return (add 함수 종료)

 

14-1. esp, 14h 만큼 add : esp가 ebp를 가르키게함 (메모리 축소)

14-2. ebp를 pop

14-3. return 후 종료

'C' 카테고리의 다른 글

C언어 구조체  (0) 2019.12.18
C언어 포인터 배열  (0) 2019.12.13
C 포인터 const  (0) 2019.12.12
C언어 배열과 포인터 차이점  (0) 2019.12.12
C언어 Char, int 정리  (0) 2019.12.05
Char

 

C언어에서 Char는 보통 8비트를 쓴다. 0000 0000 ~ 1111 1111까지 표현한다는 말

Only 양수로(unsigned) 포현하면 0~255(2의 8승), 음수도 표현하면(signed) -128 ~ 127까지이다.

 

보통 char의 디폴트 값은 signed 이나 그 보통은 컴파일러에 따라 다르다. 그러면 내 상황에서 signed인지 unsigned인지 어떻게 아느냐?

포큐아카데미

MIN값에서 -128이면 내 기본값은 signed이다. 왜냐 최소값이 -128이니깐.

 

 

INT

 

C언어 표준에 따르면 int값은 32bit가 아니라 최소 16비트 그리고 short 크기 이상인 정수형이다.

 

C언어에서 int는 기본 정수이다. 그래서 int를 기준으로 short, long이 존재한다. int 보다 작거나 크거나

CPU에게 '정수 처리해'라고 하면 CPU가 편하게 처리 할 수있는 크기여야 한다. 그 크기란 것은 CPU의 산술논리장치가 사용하는 기본 데이터이다. 이 데이터를 워드라고 하고 그 크기를 워드 크기라 한다. 워드 크기는 레지스터 크기랑 같다. 그래서 내가 사용하는 CPU에 따라 다르고 예전에는 16bit 컴퓨터를 많이 사용해서 최소 16비트 이상이다. 그러나 요즘은 거의 모든 컴퓨터가 32bit 이상이라 int는 그냥 32bit라고 말한다.

 

근데 내가 사용하는 PC는 64bit인데? 그래도 32bit에 머무른다. 왜냐? 너무 오랫동안 32bit로 머물러서 그 많은 코드를 다 수정할 수도 없고 32bit에서 64bit로 바꾼다고 해서 무조건 성능이 빨라지는 건 아니기 때문이다.

'C' 카테고리의 다른 글

C언어 구조체  (0) 2019.12.18
C언어 포인터 배열  (0) 2019.12.13
C 포인터 const  (0) 2019.12.12
C언어 배열과 포인터 차이점  (0) 2019.12.12
C 스택 메모리 해부(어셈블리어)  (3) 2019.12.09

+ Recent posts