월 19,800원
5개월 할부 시다른 수강생들이 자주 물어보는 질문이 궁금하신가요?
- 해결됨홍정모의 따라하며 배우는 C언어
배열과 포인터와 포인터 산술 연산의 이해
안녕하세요, 집에서 해당 강의로 입문한 독학러 입니다.제가 배열과 메모리, 포인터와 산술 연산을 제 기준으로 쉽게 풀어서 이해를 하였는데 이해한게 맞는건지 확인을 부탁드리고자 글을 작성하게 되었습니다.포인터변수의 크기포인터변수의 크기는 자료형이 무엇이든 x86(32비트)에선 4바이트, x64(64비트)에선 8바이트가 나왔습니다. 이러한 이유가 메모리주소에서 확인할 수가 있었는데 일단 x86(32비트)에선 왜 4바이트가 나오냐에 대한 생각인데 메모리 주소를 포인터 변수에 저장한다고 가정했을 때 메모리 주소도 값이잖아요?해당 값을 일단 출력해보면 아래와 같이 나옵니다.이 사진으로 생각할 수 있는건 16진수의 8자리 그리고 4바이트. 즉 4바이트가 표현할 수 있는 가짓수는 2^32 = 4,294,967,2960 ~ 4,294,967,295까지 여기서 4,294,967,295는 16진수로 FFFFFFFF. 위 사진의 자릿수와 같은 8자리다.즉 메모리 주소의 가짓수는 unsigned int과 마찬가지로 4,294,967,296만큼 가질 수 있으며 00000000 ~ FFFFFFFF 이기 때문에 x86(32비트)에선 포인터 변수는 해당 16진수의 값을 저장해야 하니 4바이트인 것이다. x64(64비트)에서도 위와 동일한 이유로 8바이트가 되는 것이다, 라고 이해했습니다. 맞게 이해했을까요??배열과 포인터의 산술연산배열과 포인터의 산술연산을 보고 제일 먼저 떠오른게 "주사위 게임"이였습니다.일단 배열을 선언하면 1차든 2차배열이든 몇겹이든 메모리 공간에 1차원적으로 차례대로 나열되어 메모리 공간이 확보되는걸 선생님 강의를 통해 이해했는데요. 이 메모리를 주사위 게임의 게임판 처럼 변형하여 생각을 해본다면 아래와 같이 될 것 같습니다.<- 배열의 메모리공간 가정주사위 놀이판을 편집을 해왔습니다.여기서 위 그림대로 배열을 선언한다 가정했을 때 arr[3][5]가 될 것 입니다.1~5까진 arr[0][0] ~ arr[0][4]6~10까진 arr[1][0] ~ arr[1][4]11~15까진 arr[2][0] ~ arr[2][4]자료형을 적지않은 이유는 1바이트 자료형이든 4바이트 자료형이든 저 위 그림 한칸 한칸을 하나의 인덱스라고 생각해주시면 될 것 같습니다.만약 int 4바이트라고 가정한다면 위 그림 1번칸엔 메모리 주소 0~3까지 있을 것이고 2번칸엔 4~7까지 있다고 생각할 수 있습니다.그러면 이제 저 주사위놀이판 위에 이제 말을 배치해야 인덱스칸에 접근할 수 있다고 생각해봅시다.그럼 그 말은 포인터변수가 될 것입니다.*ptr = arr; 여기서 arr은 &arr[0][0] 즉 메모리의 첫주소 값을 가지고 있기 때문에 이게 몇차원 배열이든 배열변수의 이름만 적어서 포인터변수에 첫 주소를 저장합니다.그러면 이제 1번칸에 말이 접속하여 인덱스에 포인터변수로 접근할 수 있게 된다고 생각했습니다.1번칸은 [0][0] 여기서 5번칸 [0][4]으로 이동한다고 가정 한다면 4칸을 이동해야 합니다.그러면 여기서 말에다가 4칸을 더하면 ptr += 4;을 하면 포인터변수의 주소는 [0][4]을 가리키고 있을 것 입니다. 여기서 바꿔 생각한다면[0][4]가 아니라 [0][i] 라고 했을 때 ptr += i; 을 더하면 [0][0] ~ [0][i] 까지 이동할 수 있을 것 입니다.위 방법대로 한다면 1차원이든 2차원이든 3차원이든 배열 맨 끝 [인덱스]에 포인터 변수로 접근할 수 있다고 생각합니다.그러면 이제 1번칸 [0][0]에서 6번칸 [1][0]에 가고 싶다고 가정을 한다면 총 5칸을 가야 도달할 수 있습니다.왜냐하면 arr[3][5]으로 선언했기 때문에 뒤에 [5]인덱스가 0~4까지 총 5가짓수이기 때문입니다.그러면 포인터변수(말)에게 ptr+=1*5을 하면 6번칸 [1][0]에 도달할 수 있습니다.1번칸 [0][0]에서 11번칸 [2][0]에 갈려면 ptr += 2*5을 하면 됩니다. 즉 이걸 바꿔서 생각해본다면[0][0]에서 [j][0]로 가기 위해서는 ptr += j * 5(인덱스 총 갯수)을 이용하면 옮겨갈 수 있습니다.여기서 빼기도할 수 있는데현재 ptr이 [0][0]에 있다고 가정하고 ptr += 2 * 5를 하면 [2][0]으로 가게 되는데 여기서 [1][0]으로 간다고 가정하면 ptr += -1 * 5를 하면 [1][0]으로 가게 된다. 위와 같은 방식으로 배열에 포인터변수로 접근하여 사용하니까 아주 쉽게 사용이 되더라구요, 반복문에서는 따로 위 그림처럼 포인터변수 = 말, 배열 = 주사위게임판 이라 생각하고 머릿속에 그리면서 하니까 원하는 값을 얻을 수 있고 이러한 방식이 괜찮은 방법 중 하나일까요? 아니면 잘못 이해하고 있는걸까요? 문장 정리능력이 없다보니 이렇게 긴 글을 적어놨는데 읽어주시면 감사하겠습니다.
- 미해결홍정모의 따라하며 배우는 C언어
static변수(정적변수)에 변수를 넣으면 안되나요?
강의 15:16 처럼3차원 배열의 인덱스 c + col r + (col row) *d)를함수를 통해 사용해보려 했습니다.#include <stdio.h> #include <stdlib.h> int row = 3, col = 2, depth = 2; int idx3(int c, int r, int d) { static const cr = col * row; return c + col * r + cr * d; } int main() { int* ptr = (int*)malloc(sizeof(int) * row * col * depth); if (!ptr) exit(1); for (int d = 0; d < depth; d++) for (int r = 0; r < row; r++) for (int c = 0; c < col; c++) ptr[idx3(c, r, d)] = idx3(c, r, d); for (int d = 0; d < depth; d++) { for (int r = 0; r < row; r++) { for (int c = 0; c < col; c++) printf("%d", ptr[idx3(c, r, d)]); printf("\n"); } printf("\n"); } return 0; } 실행시켰더니 expression must have a constant value라고 뜨면서 에러가 납니다.표현식은 상수를 가져야한다는 뜻인것같습니다.Q1. static변수 cr에 col *row를 넣었기때문인것같은데 원인이 무엇인가요?정적변수에는 변수를 넣으면 안되는건가요? Q2. 또 강의 15:30에서 static을 통해 cr을 사용하면 속도가 빨라지는 이유는 정적 변수 cr은 프로그램이 끝날 때까지 계속 메모리에 남아있으므로 초기화를 1번만 해주기 때문에 이후에는 연산을 할 필요가 없어서 그런것 맞나요?감사합니다!!!
- 해결됨홍정모의 따라하며 배우는 C언어
9-18. 포인터형 매개변
안녕하세요!강의 마지막 코드를 제가 제대로 이해했는지 궁금하여 질문 드립니다. #define CRTSECURE_NO_WARNINGS#include <stdio.h>void swap(int* u, int* v){printf("%p %p\n", u, v);int temp = *u;u = v;*v = temp;}int main(){int a = 123;int b = 456;printf("%p %p\n", &a, &b);swap(&a, &b);printf("%d %d\n", a, b);return 0;} a와 b의 주소 값이 매개 변수의 값으로서 함수 swap()에 입력된다.즉 int* u = &a가 되고, int* v = &b가 된다.swap() 함수의 변수 u는 main() 함수의 변수 a의 주소가 저장된 포인터 변수가 됐고, v또한 마찬가지다. u, v에 각각 a와 b의 주소 값이 저장됐기 때문에 이것을 실행하면 main() 함수의 a, b의 주소 값과 동일하게 출력된다.이후 변수 temp에 변수 a의 주소를 redirection 하여, 해당 주소 안에 저장된 값을 대입한다(a의 주소로 접근 -> 주소 안에 있는 값 123 temp에 대입).a의 주소로 직접 접근하여 그 주소의 값을 b의 주소 안에 저장된 값으로 변경한다.이후 b의 주소로 직접 접근하여 그 주소의 값을 변수 temp의 저장된 값으로 변경한다.swap() 함수에서 a, b의 주소로 직접 접근해 값을 변경했기 때문에 함수들의 영역이 달라도 main() 함수 변수들의 값은 정상적으로 swap 된다. 감사합니다.
- 미해결홍정모의 따라하며 배우는 C언어
오류 이유가 궁금합니다
int main(){ int count = 0; while (1) { printf("Current count is %d. Continue?(y/n)\n", count); if (getchar() == 'n') break; else if (getchar() == 'y') count++; else printf("Please input y or n\n"); while (getchar() != '\n') continue; } return 0;} 강의에서 int c = getchat() 해서 c를 사용하니까 잘 되던데, if문이랑 else if문에 getchar()을 그대로 넣으면 왜 오류가 나는지 궁금합니다!
- 미해결홍정모의 따라하며 배우는 C언어
예제 풀이 이렇게 하면 틀린 건가요?
while문을 한번만 써서int main(){ const int secret_code = 337; int guess = 0; while (guess != secret_code) { printf("Enter secret code :"); scanf("%d", &guess); } printf("Great!"); return 0;} 이렇게 쓰니까 결과는 잘 나오던데 틀린 방법인가요?아니면 for문으로int main(){ for (int code = 0;code != 337;printf("Enter secret code :"), scanf("%d", &code)) ; printf("Great!"); return 0;} 이렇게 하는 것도 안되나요?? 잘 돌아가긴 해서 질문 남깁니다!
- 미해결홍정모의 따라하며 배우는 C언어
5.5 나누기 연산자
안녕하세요. 강의 뒷 부분의 설명을 듣고 자료형이 다른 것끼리는 곱셈이나 나눗셈을 할 수 없지만 컴파일러가 내부에서 자료형을 맞추어주어 제대로 연산 결과가 출력된다고 이해했습니다.그래서 printf("%f\n", 9.0 / 4); 를 하였을 때 4라는 int형이 double로 변환되어 2.250000이라는 결과가 출력됩니다.그런데 printf("%d\n", 9.0 / 4); 로 하여 출력해보니 0이라는 결과가 나옵니다. 설명대로라면 %d로 바꾸었을 때 2가 출력되어야 하는것이 아닌가요??
- 미해결홍정모의 따라하며 배우는 C언어
13.6 파일 임의 접근
이번 강의 부분에서cur =ftell(fp); 부분에서 자꾸 Breakpoint Instruction Executed라고 뜨면서 오류가 납니다. 아래 결과창을 보면 fp에 null값이 들어간다고 나오는데 어떻게 해결해야 할까요?
- 미해결홍정모의 따라하며 배우는 C언어
4.2 sizeof 연산자
강의 마지막에 구조체를 활용한 크기 계산에 대해서 설명해주신 부분에 대해서 궁금한 점이 생겨서 질문 남깁니다.이렇게 하면,콘솔창에서 8이 출력됩니다.다른 자료형도 잘 되는지 궁금해서 float형을 double형으로 바꾸었더니 결과가 16이 나옵니다. double 형은 8bytes이고 int형은 4bytes이니까 12bytes가 나와야 하는 것이 아닌가요??
- 미해결홍정모의 따라하며 배우는 C언어
2.9 함수 만들기 void 질문
2.9 함수 만들기에서 선생님이 이렇게 입력하셨는데요 왜 void를 써야 하는지, void 함수의 역할이 자세하게 궁금합니다. 그리고 아래 괄호 사이에 printf("Hello, World!\n"); 를 왜 입력해야 하는지 모르겠습니다.
- 미해결홍정모의 따라하며 배우는 C언어
fgets로 입력받은 문자열에 남는 부분
강의 14:30 코드를 치고 디버거로 확인해봤는데교수님의 경우 hello다음 \n이 있고나머지는 다 \0로 채워져있는데'?'가 채워져있네요. 왜 이럴까요?gets()함수로 문자열을 입력받은 후 배열의 나머지 자리에는 '\0'으로 채워지나요?또 gets()말고 scanf()든 fgets()도 '\0'로 채워지나요?scanf(),fgets()도 디버거로 확인해보면 다 '?'로 뜨네요.
- 미해결홍정모의 따라하며 배우는 C언어
13:44 질문 있습니다
float arr2d[2][4] = { {1.0f,2.0f,3.0f,4.0f},{5.0f,6.0f,7.0f,8.0f} }; float(*pa)[4]; float* ap[2]; pa = arr2d; ap[0] = arr2d[0]; ap[1] = arr2d[1]; printf("%p %p\n", ap, (ap + 1));이 부분에서 출력이 ap변수 본인의 주소가 출력되는건 이해 했습니다 . int arr[2][3] = { {1,2,3},{4,5,6} }; int* parr[2]; parr[0] = arr[0]; parr[1] = arr[1]; for (int j = 0; j < 2; ++j) { for (int i = 0; i < 3; ++i) printf("%d %d %d %d\n", arr[j][i], parr[j][i], *(parr[j] + i), *(*(parr + j) + i)); printf("\n"); }이 코드는 10.13강의 13:16초 부분에서 가져온 코드인데요 . 궁금한게 이중포문 안쪽 부분에서 *(*(parr + j) + i) <-- 얘가 계산될때 괄호에 의해서 (parr + j) 가 먼저 연산이 되잖아요 ?? 그러면 이때도 parr 본인의 주소값에 j가 더해져서 엉뚱한값이 나와야 할텐데 , 정상적으로 출력이 되어서 위에 ap변수가 본인의 주소값을 출력할때랑 어떤부분이 다른지 궁금합니다 ...
- 미해결홍정모의 따라하며 배우는 C언어
2차원배열이 함수에 들어왔을때
#include <stdio.h> #define ROWS 3 #define COLS 4 int sum2d_3(int* arr, int rows, int cols); int main() { int data[ROWS][COLS] = { {1,2,3,4},{5,6,7,8},{9,0,1,2} }; return 0; } int sum2d_3(int *arr, int rows, int cols) { int r, c, tot = 0; for (r = 0; r < rows; r++) for (c = 0; c < cols; c++) tot += *(arr + c + cols * r); return tot; }여기서 data는 포인터 연산시 자료형이 int(*)[4] 주소자료형입니다.2차원 배열 data를 함수의 안에 넣고 매개변수 int *arr로 사용할때 tot += *(arr + c + cols * r); 를 보면 마치 배열 arr가 1차원 배열인것처럼 사용되고 있습니다. Q.2차원 배열은 함수안에 들어가면 1차원배열처럼 포인터 연산이 되는건가요? 감사합니다.
- 미해결홍정모의 따라하며 배우는 C언어
int (*pa)[4]와 int*pa의 차이
#include <stdio.h> int main() { float arr2d[2][4] = { {1.0f,2.0f,3.0f,4.0f},{5.0f,6.0f,7.0f,8.0f} }; float(*pa)[4] = arr2d; printf("%u %u \n", (unsigned)pa, (unsigned)(pa + 1)); printf("%u %u\n", (unsigned)arr2d[0], (unsigned)arr2d[1]); printf("%u %u\n", (unsigned)pa[0], (unsigned)(pa[0] + 1)); printf("%f\n", pa[0][0]); printf("%f\n", *pa[0]); printf("%f\n", **pa); printf("%f\n", pa[1][3]); printf("%f\n", *(*(pa + 1) + 3)); float* ptr = arr2d; printf("%u %u \n", (unsigned)ptr, (unsigned)(ptr + 1)); printf("%u %u\n", (unsigned)arr2d[0], (unsigned)arr2d[1]); printf("%u %u\n", (unsigned)ptr[0], (unsigned)(ptr[0] + 1)); printf("%f\n", ptr[0][0]); printf("%f\n", *ptr[0]); printf("%f\n", **ptr); printf("%f\n", ptr[1][3]); printf("%f\n", *(*(ptr + 1) + 3)); }강의 9:47에서 int(*pa)[4]가 나옵니다.강의 10:20에서 교수님께서 pa는 4개의 float자료형을 가진 배열에 대한 포인터라고 하셨습니다.저는 이것을 [4개의 float자료형을 가진 배열]의 배열에 대한 즉, 2차원배열에 대한 포인터라고 이해했습니다.float arr[4] = { 1.0f,2.0f,3.0f, 4.0f }; float(*pa)[4] = arr;Q1. 위처럼 (*pa)[4]에 1차원 배열도 넣을수 있는건가요? float(*pa)[4]에서 [4]가 헷갈리는것같습니다. 정확히 정의를 내리기가 힘듭니다. float(*pa)[4]란 무엇인가요?ㅡㅡㅡㅡㅡㅡㅡㅡㅡㅡㅡㅡㅡㅡㅡㅡㅡㅡㅡㅡㅡㅡpa는 어쨋든 그냥 단순한 포인터 변수 하나 라고 이해했습니다.그래서 똑같이float* ptr = arr2d; 포인터 변수를 하나 선언하고 거기에 2차원 배열([4개의 float자료형을 가진 배열]의 배열을 넣어줬습니다.저는 pa랑 ptr은 그냥 외형만 다를뿐 문법적으로 같은 포인터변수라고 이해했습니다.그런데 pa같은 경우printf("%f\n", pa[0][0]); printf("%f\n", *pa[0]); printf("%f\n", **pa); printf("%f\n", pa[1][3]); printf("%f\n", *(*(pa + 1) + 3));이중포인터처럼 사용가능한 반면ptr의 경우는printf("%f\n", ptr[0][0]); printf("%f\n", *ptr[0]); printf("%f\n", **ptr); printf("%f\n", ptr[1][3]); printf("%f\n", *(*(ptr + 1) + 3));이중포인터처럼 사용하면 밑줄이 뜨고 오류가 떳습니다.그림을 그려보니 ptr은 당연히 단순 포인터변수이기에 이중포인터처럼 사용은 불가능했습니다. Q2.그렇다면 어째서 pa는 포인터변수인데도 이중포인터처럼 사용이 가능한건가요? 또 pa와 ptr의 차이는 무엇인가요? 항상 감사합니다. 진짜 진짜 진짜 감사합니다.
- 미해결홍정모의 따라하며 배우는 C언어
2차원 배열에서의 주소
#include <stdio.h> int main() { int arr[2][3] = { {1,2,3}, {4,5,6} }; printf("%u %u %u %u %u", arr+1 ,arr[1], &arr[1], &arr[0]+1, arr[0] + 1); return 0; }강의 4:50 화면을 보면서 궁금한게 있습니다.arr == arr[0] == &arr[0]이고arr+1 = &arr[1] == &arr[0]+1이므로arr[0]+1또한 arr+1 = &arr[1] == &arr[0]+1과 같은 주소값을 출력할거라 생각했는데 arr[0]+1을 출력하니 &arr[0][1]이 출력되었습니다.Q1. 그 이유가 무엇인가요?강의 3:18에서 arr[0]가 두번째 인덱스가 붙는 배열의 이름이라고 하셨는데 그러므로 arr[0]는 {1,2,3}의 첫번째 원소의 주소를 가리키는 포인터가 되고 arr[0] +1은 포인터연산에 의해 데이터 2의 주소를 가리키는 포인터가 되는건가요? 제가 제대로 이해한게 맞을까요?Q2. 그렇다면 2차원 배열 arr[2][3] = {{1,2,3},{4,5,6}}에서arr은 {{1,2,3},{4,5,6}}의 첫번째 원소{1,2,3}의 주소를 가리키는 포인터인데 첫번째 원소가 {1,2,3}이므로 이중포인터가 되는것이고, arr[0]는 {1,2,3}의 첫번째 원소의 주소를 가리키는 포인터가 맞나요?굉장히 헷갈리고 어지럽네요...감사합니다.
- 미해결홍정모의 따라하며 배우는 C언어
16진수에 대한 설명
<3.8 8진수와 16진수>강의에서 16진수에 대해 설명하시면서unsigned int의 최댓값을 각각 10진수, 2진수, 8진수, 16진수로 나타내주셨습니다.16진수의 표현에서 f가 8개인 이유는 16이 2의 4제곱이기 때문이라고 하셨는데 해당 설명의 메커니즘이 잘 이해가 되지 않아 질문드립니다.수강 후 공부해본 결과, 2진수 1111이 16진수에서 f 하나에 해당하기 때문에, 해당 unsigned int의 최댓값인 2진수의 32비트는 나누기 4를 하여 16진수에서 8자리가 나타난다는 것은 이해를 하였습니다.그러나 16이 2의 4제곱이라서 8개의 자리만 있으면 된다는 설명에 대해선 이해가 어려워 조금 더 구체적으로 알려주시면 감사하겠습니다.
- 미해결홍정모의 따라하며 배우는 C언어
왜 int, long, long long 타입의 변수에 최솟값을 할당하면 언더플로우가 아니고 에러가 발생하나요?
오버플로우를 확인해보고 언더플로우도 확인해보려고 최솟값을 할당한 후 값에 -1을 할 계획이었습니다.그런데 최솟값을 할당하는 것 자체가 에러가 발생했습니다.에러는 error C4146로https://learn.microsoft.com/ko-kr/cpp/error-messages/compiler-warnings/compiler-warning-level-2-c4146?view=msvc-170위의 사이트에서 설명을 참고했습니다.그런데 위 사이트의 설명에 의한다면 char과 short도 최솟값이 할당이 안되고 에러가 나야할 것 같은데 char과 short는 최솟값도 할당이 잘 되고, 최솟값에 -1을 했을 때 언더플로우도 잘 발생(?)되었습니다.혹시 왜 그런것인지 알려주실 수 있으실까요..?!
- 미해결홍정모의 따라하며 배우는 C언어
gcc main.c 컴파일 안됨
아예 다 지우고 다시 해봤는데도 gcc main.c 를 쳐도 오류만 뜹니다. shell창이 저는 cmd로 되어있는 것과는 관련이 없을까요? F1을 쳐서 shell을 쳐보았는데 강의에 나오는 것처럼 shell 종류를 바꿀 수 있는 것이 저는 뜨지 않더라고요. 전에 파이썬 배울 때 설정해 두었던 것이라서 어떻게 했는지 잘 기억이 안나요. 저것도 powershell로 바꾸는 방법도 알려주시면 감사할 것 같습니다.
- 미해결홍정모의 따라하며 배우는 C언어
1.13 비주얼 스튜디오 코드의 기본적인 사용 방법 5분 48초쯤
선생님 따라서 .\a.exe를 쳤는데 왜 오류가 뜨는건지 모르겠어요
- 미해결홍정모의 따라하며 배우는 C언어
1.11 명령 프롬포트의 기본적인 사용법 2분 42초쯤
선생님이 보여주신 명령 프롬포트 화면이랑 제 노트북에 뜨는거랑 달라서요. explorer를 어디다가 붙여야 하는지 모르겠습니다.. 알려주세요
- 미해결홍정모의 따라하며 배우는 C언어
arr2d[2][4] 질문 있습니다
float arr2d[2][4]; 가 있을 때 arr2d의 사이즈 자체는 32바이트이지만 arr2d는 메모리공간은 없는 float 4개짜리 배열의 포인터잖아요.제가 잘못 배운 게 아니라면 뭔가 괴리감이 크게 느껴집니다. 그냥 받아들일까요?