標籤:左值 nbsp 活性 是什麼 接下來 函數 情況 方法 遍曆
指標是一個特殊的變數,它裡面儲存的數值被解釋成為記憶體裡的一個地址。 要搞清一個指標需要搞清指標的四方面的內容:指標的類型,指標所指向的 類型,指標的值或者叫指標所指向的記憶體區,還有指標本身所佔據的記憶體區。讓我們分別說明。
先聲明幾個指標放著做例子:
例一:
(1)int*ptr;
(2)char*ptr;
(3)int**ptr;
(4)int(*ptr)[3];
(5)int*(*ptr)[4];
指標的類型
從文法的角度看,你只要把指標聲明語句裡的指標名字去掉,剩下的部分就是這個指標的類型。這是指標本身所具有的類型。讓我們看看例一中各個指標的類型:
(1)int*ptr;//指標的類型是int*
(2)char*ptr;//指標的類型是char*
(3)int**ptr;//指標的類型是int**
(4)int(*ptr)[3];//指標的類型是int(*)[3]
(5)int*(*ptr)[4];//指標的類型是int*(*)[4]
怎麼樣?找出指標的類型的方法是不是很簡單?
指標所指向的類型
當你通過指標來訪問指標所指向的記憶體區時,指標所指向的類型決定了編譯器將把那片記憶體區裡的內容當做什麼來看待。
從文法上看,你只須把指標聲明語句中的指標名字和名字左邊的指標聲明符*去掉,剩下的就是指標所指向的類型。例如:
(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越來越熟悉時,你會發現,把與指標攪和在一起的 "類型 "這個概念分成 "指標的類型 "和 "指標所指向的類型 "兩個概念,是精通指標的關鍵點之一。我看了不少書,發現有些寫得差的書中,就把指標的這兩個概念攪在一起了,所以看起書來前後矛盾,越看越糊塗。
指標的值,或者叫指標所指向的記憶體區或地址
指標的值是指標本身儲存的數值,這個值將被編譯器當作一個地址,而不是一個一般的數值。在32位程式裡,所有類型的指標的值都是一個32位整數,因為32位程式裡記憶體位址全都是32位長。 指標所指向的記憶體區就是從指標的值所代表的那個記憶體位址開始,長度為si zeof(指標所指向的類型)的一片記憶體區。以後,我們說一個指標的值是XX,就相當於說該指標指向了以XX為首地址的一片記憶體地區;我們說一個指標指向了某塊記憶體地區,就相當於說該指標的值是這塊記憶體地區的首地址。
指標所指向的記憶體區和指標所指向的類型是兩個完全不同的概念。在例一中,指標所指向的類型已經有了,但由於指標還未初始化,所以它所指向的記憶體區是不存在的,或者說是無意義的。
以後,每遇到一個指標,都應該問問:這個指標的類型是什嗎?指標指的類型是什嗎?該指標指向了哪裡?
指標本身所佔據的記憶體區
指標本身佔了多大的記憶體?你只要用函數sizeof(指標的類型)測一下就知道了。在32位平台裡,指標本身佔據了4個位元組的長度。
指標本身佔據的記憶體這個概念在判斷一個指標運算式是否是左值時很有用。
指標的算術運算
指標可以加上或減去一個整數。指標的這種運算的意義和通常的數值的加減運算的意義是不一樣的。例如:
例二:
1、chara[20];
2、int*ptr=a;
...
...
3、ptr++;
在上例中,指標ptr的類型是int*,它指向的類型是int,它被初始化為指向整形變數a。接下來的第3句中,指標ptr被加了1,編譯器是這樣處理的:它把指標ptr的值加上了sizeof(int),在32位程式中,是被加上了4。由於地址是用位元組做單位的,故ptr所指向的地址由原來的變數a的地址向高地址方向增加了4個位元組。
由於char類型的長度是一個位元組,所以,原來ptr是指向數組a的第0號單元開始的四個位元組,此時指向了數組a中從第4號單元開始的四個位元組。
我們可以用一個指標和一個迴圈來遍曆一個數組,看例子:
例三:
intarray[20];
int*ptr=array;
...
//此處略去為整型數組賦值的代碼。
...
for(i=0;i <20;i++)
{
(*ptr)++;
ptr++;
}
這個例子將整型數組中各個單元的值加1。由於每次迴圈都將指標ptr加1,所以每次迴圈都能訪問數組的下一個單元。
再看例子:
例四:
1、chara[20];
2、int*ptr=a;
...
...
3、ptr+=5;
在這個例子中,ptr被加上了5,編譯器是這樣處理的:將指標ptr的值加上5乘sizeof(int),在32位程式中就是加上了5乘4=20。由於地址的單位是位元組,故現在的ptr所指向的地址比起加5後的ptr所指向的地址來說,向高地址方向移動了20個位元組。在這個例子中,沒加5前的ptr指向數組a的第0號單元開始的四個位元組,加5後,ptr已經指向了數組a的合法範圍之外了。雖然這種情況在應用上會出問題,但在文法上卻是可以的。這也體現出了指標的靈活性。
C語言指標理解,記下來以後看