쓰려했는데 잘 쓴 글이 여럿 있네요. 그냥 요거 리터럴부터 읽으시길.


https://modoocode.com/33





제가 이 글을 읽다가 헷갈리던 부분을 서술하겠습니다.


char str[] = "Hello";


위 문장은 str에 "Hello" literal의 주소를 넣은 것으로 생각되고, 그러면 수정이 불가능한데 어째서 수정이 가능할까요?





char str[] = "Hello"; 는 널문자를 포함해서 6개의 일차원 배열이 할당됩니다.

그 안에 문자가 H, e, l, l, o, \0이 이렇게 6개가 들어가는 거죠.


즉, char str[] = "Hello";는


char arr[] = {72, 101, 108, 108, 111, 0};

char* str = arr; //arr은 배열의 이름, 주소


이와 같은 원리로 이해할 수 있습니다.

배열은 변수의 영역이므로 수정 가능하고요 :)

큰 따옴표와 작은 따옴표의 차이점?

⇒ 큰 따옴표(쌍 따옴표)는 "여러", 작은 따옴표(홑 따옴표)는 '홑'이다.

 

 

예시를 들자면

ex) 에러가 발생하는 경우

 

1)

char str[] = "HELLO";

str[0] = "B";

 

2) printf("%c","A"); printf("%s", 'A');

 

ex) 올바른 사용

 

1)

char str[] = "HELLO";

str[0] = 'B';

 

2) printf("%c", 'A'); printf("%s", "A");

 

 

자세히 설명하면 다음과 같다

 

 In C and in C++ single quotes identify a single character, while double quotes create a string literal. 'a' is a single a character literal, while "a" is a string literal containing an 'a' and a null terminator (that is a 2 char array). //stack overflow 참고

 

 

즉, 우선 한 개는 홑따옴표, 여러 개는 쌍따옴표가 맞다. 그러나 한 개를 표현할때도 쌍따옴표를 쓰면 뒤에 널문자가 붙는다는 것!

따라서 널문자가 나타날때까지 읽는 %s에서는 한 문자를 쓰더라도 쌍따옴표(double quotations)를 사용해야하는 것이다.

※읽다보면 헷갈리는 부분이 있을 수 있습니다. 그럴 때 거기에 막혀서 검색해보기 전에, 일단 모두 읽어보시길 추천합니다!
서술된 내용 뒤에 보충 설명을 달아놓은 경우가 많습니다^^

 

 

 

 

포인터란?


 


C에서 포인터 변수는

자료형* 변수명;                 <- 이렇게 선언되고는 합니다.

 

※포인터 변수 선언 방식

1. 자료형* 변수명;

2. 자료형 * 변수명;

3. 자료형 *변수명;

모두 가능합니다.

그러나 저는 1번 방식을 추천드립니다

 

 


저는 포인터=주소명, *포인터=주소에 도착한 포인터 라고 이해했습니다.



다음의 간단한 예제를 보며 이해를 해보겠습니다.(이해에 앞서, *연산자는 주소에 접근, &연산자는 주소를 반환합니다.)

int main(void) {

    int num = 5;
    int* ptr; //int형 포인터 변수 ptr 선언

    ptr = &num; //포인터 변수 ptr에 num의 주소를 저장(&num은 num의 주소를 반환합니다)



    printf("주소명(포인터) = 0x%x \n", ptr); //주소명 표시
    printf("주소에 도착한 포인터(*포인터) = %d \n", *ptr); //주소에 도착한 포인터 변수 ptr이 마치 num인듯이 행동

    return 0;
}


실행결과



주소를 16진수로 표현하는 컴퓨터 업계에서 주로 컴퓨터의 언어인 2진수로 쉽게 표현할 수 있고, 2진수를 줄여서 쓸 수 있기 때문이라 합니다. 




좀 더 자세히 설명하자면(사실상 저는 위의 예제 하나면 포인터의 개념을 모두 설명했다 생각합니다.)


포인터 변수란 메모리의 주소 값을 저장하기 위한 변수입니다.

&연산자를 통해 num의 주소 값을 반환하여,

int형 포인터 변수 ptr에는 변수 num의 '시작번지 주소'가 저장됩니다.

그리고 *연산자를 이용해 포인터가 가르키는 메모리 공간에 접근합니다.(*ptr을 통해 num이 가르키는 메모리에 접근하여 num처럼 행동하였습니다)


이러면 포인터 개념 설명 끝!


+추가)


포인터 변수에도 다양한 타입이 존재합니다. int, double, char, unsigned int 등등..

포인터 변수에 이러한 타입이 왜 존재하느냐!하면 다음 코드를 보고 생각해봅시다.


double num = 3.14;
int* pnum=&num;


double형의 크기는 8byte, int형의 크기는 4byte입니다.

포인터 변수 pnum는 int형이고 크기는4byte인데 여기에는 double형 num의 주소가 담겨있습니다. 이는

 "num의 시작 주소부터 4byte를 읽겠다"를 의미합니다.

허나 num의 크기는 8byte이므로 포인터 변수 pnum은 *pnum으로 num에 접근하였을 때 읽다말아서(8byte 중에서 4byte만 읽어서) 원하는 값을 표시하지 않습니다.


따 라 서~~ 포인터 변수의 타입의 존재 이유는 * 연산자로 메모리를 올바르게 읽기 위해서라 할 수 있을것 같습니다.


++추추가)


잘못된 포인터의 사용과 널 포인터


다음의 포인터 선언의 잘못된 예를 보시고 무엇이 잘못되었는지 맞춰보시기 바랍니다.


int* ptr;
*ptr = 0;




C언어에서 변수를 초기화하지 않고 선언 시, 쓰레기값이 저장됩니다.

int *ptr; 을 실행하였을 때, 포인터 변수 ptr에는 임의의 주소값(쓰레기 값)이 저장될 것 입니다.

*ptr = 0;을 실행한다면? 그 임의의 주소값에 0을 저장하겠죠.


근데 만약 *ptr이 가르키는 곳이 메모리상의 중요한 부분이였다면 어떻게 될까요?? 컴퓨터를 다신 사용할 수 없을지도 모릅니다..

다행히 요즘 운영체제는 이런 사고를 방지시켜준다 하네요.

그렇다면 올바르게 사용하려면 어떻게 해야할까요??



int* ptr = &num; //<- 처럼 선언과 동시에 초기화 시켜주던가,

int* ptr = 0;
int* ptr = NULL;  //<- 처럼 주소값에 0또는 NULL을 저장합니다.

이는 0번지를 의미하는 것이 아닌, 아무것도 가리키지 않는다를 의미한답니다.

 

 

포인터 설명을 위한 포스팅은 계속 됩니다. 다음 포스팅을 참고하시길 바랍니다

 

 

 

 

이 포스터는 윤성우 열혈 C 프로그래밍을 참고하여 작성하였습니다.

 

궁금한 점은 댓글로 물어보시길 바랍니다.

부족한 점이 많으니 태클 환영합니다!




+ Recent posts