1、指針的概念
指針是壹個特殊的變量,它裏面存儲的數值被解釋成為內存裏的壹個地址。要搞清壹個指針需要搞清指針的四方面的內容:指針的類型,指針所指向的類型,指針的值或者叫指針所指向的內存區,還有指針本身所占據的內存區。分別說明。
先聲明幾個指針放著做例子:
int *ptr;?
char *ptr;?
int **ptr;?
int (*ptr)[3];?
int *(*ptr)[4];?
2、指針的類型
從語法的角度看,只要把指針聲明語句裏的指針名字去掉,剩下的部分就是這個指針的類型。這是指針本身所具有的類型。看看下面各個指針的類型:?
int *ptr; //指針的類型是int *?
char *ptr; //指針的類型是char *?
int **ptr; //指針的類型是 int **?
int (*ptr)[3]; //指針的類型是 int(*)[3]?
int *(*ptr)[4]; //指針的類型是 int *(*)[4]?
3、指針所指向的類型
當通過指針來訪問指針所指向的內存區時,指針所指向的類型決定了編譯器將把那片內存區裏的內容當做什麽來看待。從語法上看,只須把指針聲明語句中的指針名字和名字左邊的指針聲明符*去掉,剩下的就是指針所指向的類型。例如:?
int *ptr; //指針所指向的類型是int?
char *ptr; //指針所指向的的類型是char?
int **ptr; //指針所指向的的類型是 int *?
int (*ptr)[3]; //指針所指向的的類型是 int()[3]?
int *(*ptr)[4]; //指針所指向的的類型是 int *()[4]?
在指針的算術運算中,指針所指向的類型有很大的作用。?
指針的類型(即指針本身的類型)和指針所指向的類型是兩個概念。當對C越來越熟悉時,會發現,把與指針攪和在壹起的“類型”這個概念分成“指針的類型”和“指針所指向的類型”兩個概念,是精通指針的關鍵點之壹。
5、指針的值
指針的值是指針本身存儲的數值,這個值將被編譯器當作壹個地址,而不是壹個壹般的數值。在32位程序裏,所有類型的指針的值都是壹個32位整數,因為32位程序裏內存地址全都是32位長。?
指針所指向的內存區就是從指針的值所代表的那個內存地址開始,長度為sizeof(指針所指向的類型)的壹片內存區。以後,我們說壹個指針的值是XX,就相當於說該指針指向了以XX為首地址的壹片內存區域;壹個指針指向了某塊內存區域,就相當於說該指針的值是這塊內存區域的首地址。?
指針所指向的內存區和指針所指向的類型是兩個完全不同的概念。在例壹中,指針所指向的類型已經有了,但由於指針還未初始化,所以它所指向的內存區是不存在的,或者說是無意義的。?
6、指針本身所占據的內存區
指針本身占了多大的內存;只要用函數sizeof(指針的類型)測壹下就知道了。在32位平臺裏,指針本身占據了4個字節的長度。
指針本身占據的內存這個概念在判斷壹個指針表達式是否是左值時很有用。
擴展資料:
指針的算術運算
指針可以加上或減去壹個整數。指針的這種運算的意義和通常的數值的加減運算的意義是不壹樣的。例如:?
例壹:?
char a[20];?
int *ptr=a;?
...?
...?
ptr++;?
在上例中,指針ptr的類型是int*,它指向的類型是int,它被初始化為指向整形變量a。接下來的第3句中,指針ptr被加了1,
編譯器是這樣處理的:它把指針ptr的值加上了sizeof(int),在32位程序中,是被加上了4。由於地址是用字節做單位的,故ptr所指向的地址由原來的變量a的地址向高地址方向增加了4個字節。
由於char類型的長度是壹個字節,所以,原來ptr是指向數組a的第0號單元開始的四個字節,此時指向了數組a中從第4號單元開始的四個字節。
可以用壹個指針和壹個循環來遍歷壹個數組,看例子:?
例二:?
這個例子將整型數組中各個單元的值加1。由於每次循環都將指針ptr加1,所以每次循環都能訪問數組的下壹個單元。再看例子:?
例三:?
char a[20];?
int *ptr = a;?
...?
...?
ptr += 5;?
在這個例子中,ptr被加上了5,編譯器是這樣處理的:將指針ptr的值加上5乘sizeof(int),在32位程序中就是加上了5乘4=20。由於地址的單位是字節,故現在的ptr所指向的地址比起加5後的ptr所指向的地址來說。
向高地址方向移動了20個字節。在這個例子中,沒加5前的ptr指向數組a的第0號單元開始的四個字節,加5後,ptr已經指向了數組a的合法範圍之外了。雖然這種情況在應用上會出問題,但在語法上卻是可以的。這也體現出了指針的靈活性。
如果上例中,ptr是被減去5,那麽處理過程大同小異,只不過ptr的值是被減去5乘sizeof(int),新的ptr指向的地址將比原來的ptr所指向的地址向低地址方向移動了20個字節。?
總結壹下,壹個指針ptrold加上壹個整數n後,結果是壹個新的指針ptrnew,ptrnew的類型和ptrold的類型相同,ptrnew所指向的類型和ptrold所指向的類型也相同。ptrnew的值將比ptrold的值增加了n乘sizeof(ptrold所指向的類型)個字節。
就是說,ptrnew所指向的內存區將比ptrold所指向的內存區向高地址方向移動了n乘sizeof(ptrold所指向的類型)個字節。壹個指針ptrold減去壹個整數n後,
結果是壹個新的指針ptrnew,ptrnew的類型和ptrold的類型相同,ptrnew所指向的類型和ptrold所指向的類型也相同。
ptrnew的值將比ptrold的值減少了n乘sizeof(ptrold所指向的類型)個字節,就是說,ptrnew所指向的內存區將比ptrold所指向的內存區向低地址方向移動了n乘sizeof(ptrold所指向的類型)個字節。
參考資料:
百度百科——指針