1. 程式人生 > >深入理解指標必看

深入理解指標必看

一。指標的概念    指標是一個特殊的變數,它裡面儲存的數值被解釋成為記憶體裡的一個地址。    要搞清一個指標需要搞清指標的四方面的內容:指標的型別,指標所指向的型別,指標的值或者叫指標所指向的記憶體區,還有指標本身所佔據的記憶體區。

1。 指標的型別    從語法的角度看,你只要把指標宣告語句裡的指標名字去掉,剩下的部分就是這個指標的型別。這是指標本身所具有的型別。讓我們看看例一中各個指標的型別:(1)int *ptr; //指標的型別是int *(2)char *ptr; //指標的型別是char *(3)int **ptr; //指標的型別是 int **(4)int (*ptr)[3]; //指標的型別是 

int(*)[3](5)int *(*ptr)[4]; //指標的型別是 int *(*)[4]怎麼樣?找出指標的型別的方法是不是很簡單?2。指標所指向的型別    從語法上看,你只須把指標宣告語句中的指標名字和名字左邊的指標宣告符 *去掉,剩下的就是指標所指向的型別。例如:(1)int *ptr; //指標所指向的型別是int(2)char *ptr; //指標所指向的的型別是char(3)int **ptr; //指標所指向的的型別是 int *(4)int (*ptr)[3]; //指標所指向的的型別是 int()[3](5)int *(*ptr)[4]; //指標所指向的的型別是 int *()[4]    在指標的算術運算中,指標所指向的型別有很大的作用。
    指標的型別(即指標本身的型別)和指標所指向的型別是兩個概念。當你對C越來越熟悉時,你會發現,把與指標攪和在一起的型別這個概念分成指標的類 型指標所指向的型別兩個概念,是精通指標的關鍵點之一。指標儲存的是一個地址,這個地址是某一塊記憶體的地址,這個地址裡放的一定是一個數據,這個資料是有型別的,我們所知,int型佔四個位元組,char就佔一個位元組等等,當你通過指標來訪問指標所指向的記憶體區時,指標所指向的型別決定了編譯器將把那片記憶體區裡的內容當做什麼來看待。int型就取四個位元組,char就取一個位元組,這個在做指標運算時,非常重要,需要特別注意。

3。 指標的值 指標的值,或者叫指標所指向的記憶體區或地址。

 指標的值是指標本身儲存的數值,這個值將被編譯器當作一個地址,而不是一個一般的數值。在32位程式裡,所有型別的指標的值都是一個32位整數,因為32位程式裡記憶體地址全都是32位長。    指標所指向的記憶體區就是從指標的值所代表的那個記憶體地址開始,長度為sizeof(指標所指向的型別)的一片記憶體區。以後,我們說一個指標的值是XX,就 相當於說該指標指向了以XX為首地址的一片記憶體區域;我們說一個指標指向了某塊記憶體區域,就相當於說該指標的值是這塊記憶體區域的首地址。    指標所指向的記憶體區和指標所指向的型別是兩個完全不同的概念。在例一中,指標所指向的型別已經有了,但由於指標還未初始化,所以它所指向的記憶體區是不存在的,或者說是無意義的。    以後,每遇到一個指標,都應該問問:這個指標的型別是什麼?指標指向的型別是什麼?該指標指向了哪裡?4。 指標本身所佔據的記憶體區。    指標本身佔了多大的記憶體?你只要用sizeof(指標的型別)測一下就知道了。在32位平臺裡,指標本身佔據了4個位元組的長度。    指標本身佔據的記憶體這個概念在判斷一個指標表示式是否是左值時很有用。二。指標的算術運算    指標可以加上或減去一個整數。指標的這種運算的意義和通常的數值的加減運算的意義是不一樣的。

  指標p的型別是int*,它指向的型別是int,它被初始化為指向整形陣列a。接下來指標p被加了1,編譯器是這樣處理 的:它把指標p的值加上了sizeof(int),在32位程式中,是被加上了4。由於地址是用位元組做 單位的,故p所指向的地址由原來的變數a的地址向高地址方向增加了4個位元組。由於int型別的長度是四個位元組,所以,原來p是指向陣列a的第號單元開始的四個位元組,此時指向了陣列a中從第一個元素。同理,指標q加1,q所指向的地址向高地址增加了1個位元組,陣列b的每一個元素是char型別的,佔1個位元組,所以此時指向了陣列的下一個元素。第三個for迴圈,實際上就是q所指向的地址加上sizeof(char),輸出的是陣列b首地址加1.“printf("%d\n",sizeof(char *));”輸出的是指標本身所佔的大小,指標本身佔據了4個位元組的長度。