詮釋C指針操作
我們可以用一個(gè)指針和一個(gè)循環(huán)來遍歷一個(gè)數(shù)組,看例子:
例三:
int
int
...
//此處略去為整型數(shù)組賦值的代碼。
...
for(i=0;i<20;i++)
{
(*ptr)++;
ptr++;
}
這個(gè)例子將整型數(shù)組中各個(gè)單元的值加1。由于每次循環(huán)都將指針
ptr加1,所以每次循環(huán)都能訪問數(shù)組的下一個(gè)單元。再看例子:
例四:
1。
2。
...
...
3。
在這個(gè)例子中,ptr被加上了5,編譯器是這樣處理的:將指針ptr
的值加上5乘sizeof(int),在32位程序中就是加上了5乘4=20。由于地址的單位是字節(jié),故
現(xiàn)在的ptr所指向的地址比起加5后的ptr所指向的地址來說,向高地址方向移動(dòng)了20個(gè)字
節(jié)。在這個(gè)例子中,沒加5前的ptr指向數(shù)組a的第0號單元開始的四個(gè)字節(jié),加5后,ptr已
經(jīng)指向了數(shù)組a的合法范圍之外了。雖然這種情況在應(yīng)用上會出問題,但在語法上卻是可以
的。這也體現(xiàn)出了指針的靈活性。
如果上例中,ptr是被減去5,那么處理過程大同小異,只不過ptr
的值是被減去5乘sizeof(int),新的ptr指向的地址將比原來的ptr所指向的地址向低地址
方向移動(dòng)了20個(gè)字節(jié)。
總結(jié)一下,一個(gè)指針ptrold加上一個(gè)整數(shù)n后,結(jié)果是一個(gè)新的指
針ptrnew,ptrnew的類型和ptrold的類型相同,ptrnew所指向的類型和ptrold所指向的類
型也相同。ptrnew的值將比ptrold的值增加了n乘sizeof(ptrold所指向的類型)個(gè)字節(jié)。就
是說,ptrnew所指向的內(nèi)存區(qū)將比ptrold所指向的內(nèi)存區(qū)向高地址方向移動(dòng)了n乘sizeof
(ptrold所指向的類型)個(gè)字節(jié)。
a一個(gè)指針ptrold減去一個(gè)整數(shù)n后,結(jié)果是一個(gè)新的指針ptrnew,
ptrnew的類型和ptrold的類型相同,ptrnew所指向的類型和ptrold所指向的類型也相同。
ptrnew的值將比ptrold的值減少了n乘sizeof(ptrold所指向的類型)個(gè)字節(jié),就是說,
ptrnew所指向的內(nèi)存區(qū)將比ptrold所指向的內(nèi)存區(qū)向低地址方向移動(dòng)了n乘sizeof(ptrold
所指向的類型)個(gè)字節(jié)。
第三章。運(yùn)算?amp;和*
這里&是取地址運(yùn)算符,*是...書上叫做"間接運(yùn)算符"。
&a的運(yùn)算結(jié)果是一個(gè)指針,指針的類型是a的類型加個(gè)*,指針?biāo)?br />向的類型是a的類型,指針?biāo)赶虻牡刂仿?,那就是a的地址。
*p的運(yùn)算結(jié)果就五花八門了??傊?p的結(jié)果是p所指向的東西,這
個(gè)東西有這些特點(diǎn):它的類型是p指向的類型,它所占用的地址是p所指向的地址。
例五:
int
int
int
int
p=&a;//&a的結(jié)果是一個(gè)指針,類型是int*,指向的類型是int,指
向的地址是a的地址。
*p=24;//*p的結(jié)果,在這里它的類型是int,它所占用的地址是p所
指向的地址,顯然,*p就是變量a。
ptr=&p;//&p的結(jié)果是個(gè)指針,該指針的類型是p的類型加個(gè)*,在
這里是int**。該指針?biāo)赶虻念愋褪莗的類型,這里是int*。該指針?biāo)赶虻牡刂肪褪侵?br />針p自己的地址。
*ptr=&b;//*ptr是個(gè)指針,&b的結(jié)果也是個(gè)指針,且這兩個(gè)指針的
類型和所指向的類型是一樣的,所以用&b來給*ptr賦值就是毫無問題的了。
**ptr=34;//*ptr的結(jié)果是ptr所指向的東西,在這里是一個(gè)指針,
對這個(gè)指針再做一次*運(yùn)算,結(jié)果就是一個(gè)int類型的變量。
第四章。指針表達(dá)式。
一個(gè)表達(dá)式的最后結(jié)果如果是一個(gè)指針,那么這個(gè)表達(dá)式就叫指針
表達(dá)式。
下面是一些指針表達(dá)式的例子:
例六:
int
int
int
pa=&a;//&a是一個(gè)指針表達(dá)式。
int
*ptr=&b;//*ptr和&b都是指針表達(dá)式。
pa=array;
pa++;//這也是指針表達(dá)式。
例七:
char
char
char
str=*parr;//*parr是指針表達(dá)式
str=*(parr+1);//*(parr+1)是指針表達(dá)式
str=*(parr+2);//*(parr+2)是指針表達(dá)式
由于指針表達(dá)式的結(jié)果是一個(gè)指針,所以指針表達(dá)式也具有指針?biāo)?br />具有的四個(gè)要素:指針的類型,指針?biāo)赶虻念愋?,指針指向的?nèi)存區(qū),指針自身占據(jù)的
內(nèi)存。
好了,當(dāng)一個(gè)指針表達(dá)式的結(jié)果指針已經(jīng)明確地具有了指針自身占
據(jù)的內(nèi)存的話,這個(gè)指針表達(dá)式就是一個(gè)左值,否則就不是一個(gè)左值。
在例七中,&a不是一個(gè)左值,因?yàn)樗€沒有占據(jù)明確的內(nèi)存。*ptr
是一個(gè)左值,因?yàn)?ptr這個(gè)指針已經(jīng)占據(jù)了內(nèi)存,其實(shí)*ptr就是指針pa,既然pa已經(jīng)在內(nèi)
存中有了自己的位置,那么*ptr當(dāng)然也有了自己的位置。
第五章。數(shù)組和指針的關(guān)系
如果對聲明數(shù)組的語句不太明白的話,請參閱我前段時(shí)間貼出的
文?lt;<如何理解c和c++的復(fù)雜類型聲明>>。
數(shù)組的數(shù)組名其實(shí)可以看作一個(gè)指針??聪吕?wbr />
例八:
int
...
...
value=array[0];//也可寫成:value=*array;
value=array[3];//也可寫成:value=*(array+3);
value=array[4];//也可寫成:value=*(array+4);
上例中,一般而言數(shù)組名array代表數(shù)組本身,類型是int
但如果把a(bǔ)rray看做指針的話,它指向數(shù)組的第0個(gè)單元,類型是int
組單元的類型即int。因此*array等于0就一點(diǎn)也不奇怪了。同理,array+3是一個(gè)指向數(shù)組
第3個(gè)單元的指針,所以*(array+3)等于3。其它依此類推。
評論