新聞中心

詮釋C指針操作(二)

作者: 時(shí)間:2016-11-28 來源:網(wǎng)絡(luò) 收藏
例九:

char*str[3]={
"Hello,thisisasample!","Hi,goodmorning.","Helloworld"

};
chars[80];
strcpy(s,str[0]);//也可寫成strcpy(s,*str);
strcpy(s,str[1]);//也可寫成strcpy(s,*(str+1));

strcpy(s,str[2]);//也可寫成strcpy(s,*(str+2));

上例中,str是一個(gè)三單元的數(shù)組,該數(shù)組的每個(gè)單元都是一個(gè)指
針,這些指針各指向一個(gè)字符串。把指針數(shù)組名str當(dāng)作一個(gè)指針的話,它指向數(shù)組的第0
號(hào)單元,它的類型是char**,它指向的類型是char*。

*str也是一個(gè)指針,它的類型是char*,它所指向的類型是char,
它指向的地址是字符串"Hello,thisisasample!"的第一個(gè)字符的地址,即H的地址。


str+1也是一個(gè)指針,它指向數(shù)組的第1號(hào)單元,它的類型是
char**,它指向的類型是char*。

*(str+1)也是一個(gè)指針,它的類型是char*,它所指向的類型是
char,它指向"Hi,goodmorning."的第一個(gè)字符H,等等。

下面總結(jié)一下數(shù)組的數(shù)組名的問題。聲明了一個(gè)數(shù)組TYPEarray
[n],則數(shù)組名稱array就有了兩重含義:第一,它代表整個(gè)數(shù)組,它的類型是TYPE[n];
第二,它是一個(gè)指針,該指針的類型是TYPE*,該指針指向的類型是TYPE,也就是數(shù)組單元
的類型,該指針指向的內(nèi)存區(qū)就是數(shù)組第0號(hào)單元,該指針自己占有單獨(dú)的內(nèi)存區(qū),注意它
和數(shù)組第0號(hào)單元占據(jù)的內(nèi)存區(qū)是不同的。該指針的值是不能修改的,即類似array++的表
達(dá)式是錯(cuò)誤的。

在不同的表達(dá)式中數(shù)組名array可以扮演不同的角色。


在表達(dá)式sizeof(array)中,數(shù)組名array代表數(shù)組本身,故這時(shí)
sizeof函數(shù)測出的是整個(gè)數(shù)組的大小。

在表達(dá)式*array中,array扮演的是指針,因此這個(gè)表達(dá)式的結(jié)果
就是數(shù)組第0號(hào)單元的值。sizeof(*array)測出的是數(shù)組單元的大小。

表達(dá)式array+n(其中n=0,1,2,....。)中,array扮演的是指
針,故array+n的結(jié)果是一個(gè)指針,它的類型是TYPE*,它指向的類型是TYPE,它指向數(shù)組
第n號(hào)單元。故sizeof(array+n)測出的是指針類型的大小。

例十:

intarray[10];

int(*ptr)[10];

ptr=&array;

上例中ptr是一個(gè)指針,它的類型是int(*)[10],他指向的類型是
int[10],我們用整個(gè)數(shù)組的首地址來初始化它。在語句ptr=&array中,array代表數(shù)組本
身。

本節(jié)中提到了函數(shù)sizeof(),那么我來問一問,sizeof(指針名稱)
測出的究竟是指針自身類型的大小呢還是指針?biāo)赶虻念愋偷拇笮??答案是前者。例如?br />

int(*ptr)[10];

則在32位程序中,有:

sizeof(int(*)[10])==4

sizeof(int[10])==40

sizeof(ptr)==4

實(shí)際上,sizeof(對象)測出的都是對象自身的類型的大小,而不是
別的什么類型的大小。

第六章。指針和結(jié)構(gòu)類型的關(guān)系

可以聲明一個(gè)指向結(jié)構(gòu)類型對象的指針。

例十一:
structMyStruct
{
inta;
intb;
intc;
}
MyStructss={20,30,40};//聲明了結(jié)構(gòu)對象ss,并把ss的三個(gè)成
員初始化為20,30和40。
MyStruct*ptr=&ss;//聲明了一個(gè)指向結(jié)構(gòu)對象ss的指針。它的類
型是MyStruct*,它指向的類型是MyStruct。
int*pstr=(int*)&ss;//聲明了一個(gè)指向結(jié)構(gòu)對象ss的指針。但是
它的類型和它指向的類型和ptr是不同的。
請問怎樣通過指針ptr來訪問ss的三個(gè)成員變量?

答案:
ptr->a;
ptr->b;
ptr->c;
又請問怎樣通過指針pstr來訪問ss的三個(gè)成員變量?

答案:
*pstr;//訪問了ss的成員a。
*(pstr+1);//訪問了ss的成員b。
*(pstr+2)//訪問了ss的成員c。
呵呵,雖然我在我的MSVC++6.0上調(diào)式過上述代碼,但是要知道,
這樣使用pstr來訪問結(jié)構(gòu)成員是不正規(guī)的,為了說明為什么不正規(guī),讓我們看看怎樣通過
指針來訪問數(shù)組的各個(gè)單元:

本文引用地址:http://m.butianyuan.cn/article/201611/322934.htm

例十二:
intarray[3]={35,56,37};
int*pa=array;
通過指針pa訪問數(shù)組array的三個(gè)單元的方法是:

*pa;//訪問了第0號(hào)單元
*(pa+1);//訪問了第1號(hào)單元
*(pa+2);//訪問了第2號(hào)單元
從格式上看倒是與通過指針訪問結(jié)構(gòu)成員的不正規(guī)方法的格式一
樣。
所有的C/C++編譯器在排列數(shù)組的單元時(shí),總是把各個(gè)數(shù)組單元存
放在連續(xù)的存儲(chǔ)區(qū)里,單元和單元之間沒有空隙。但在存放結(jié)構(gòu)對象的各個(gè)成員時(shí),在某
種編譯環(huán)境下,可能會(huì)需要字對齊或雙字對齊或者是別的什么對齊,需要在相鄰兩個(gè)成員
之間加若干"填充字節(jié)",這就導(dǎo)致各個(gè)成員之間可能會(huì)有若干個(gè)字節(jié)的空隙。
所以,在例十二中,即使*pstr訪問到了結(jié)構(gòu)對象ss的第一個(gè)成員
變量a,也不能保證*(pstr+1)就一定能訪問到結(jié)構(gòu)成員b。因?yàn)槌蓡Ta和成員b之間可能會(huì)有
若干填充字節(jié),說不定*(pstr+1)就正好訪問到了這些填充字節(jié)呢。這也證明了指針的靈活
性。要是你的目的就是想看看各個(gè)結(jié)構(gòu)成員之間到底有沒有填充字節(jié),嘿,這倒是個(gè)不錯(cuò)
的方法。


上一頁 1 2 下一頁

關(guān)鍵詞: C指針變

評論


技術(shù)專區(qū)

關(guān)閉