2.2.指向二維數(shù)組的指針
2.2.1.二維數(shù)組元素的地址
為了說(shuō)明問(wèn)題,我們定義以下二維數(shù)組:
int a[3][4]={{0,1,2,3},{4,5,6,7},{8,9,10,11}};
a為二維數(shù)組名,此數(shù)組有3行4列,共12個(gè)元素。但也可這樣來(lái)理解,數(shù)組a由三個(gè)元素組成:a[0],a[1],a[2]。而它勻中每個(gè)元素又是一個(gè)一維數(shù)組,且都含有4個(gè)元素(相當(dāng)于4列),例如,a[0]所代表的一維數(shù)組所包含的4個(gè)元素為a[0][0],a[0][1],a[0][2], a[0][3]。如圖5.所示:
┏━━━━┓ ┏━┳━┳━┳━┓
a─→ ┃ a[0] ┃─→ ┃0 ┃1 ┃2 ┃3 ┃
┣━━━━┫ ┣━╋━╋━╋━┫
┃ a[1] ┃─→ ┃4 ┃5 ┃6 ┃7 ┃
┣━━━━┫ ┣━╋━╋━╋━┫
┃ a[2] ┃─→ ┃8 ┃9 ┃10┃11┃
┗━━━━┛ ┗━┻━┻━┻━┛
圖5.
但從二維數(shù)組的角度來(lái)看,a代表二維數(shù)組的首地址,當(dāng)然也可看成是二維數(shù)組第0行的首地址。a+1就代表第1行的首地址,a+2就代表第2行的首地址。如果此二維數(shù)組的首地址為1000,由于第0行有4個(gè)整型元素,所以a+1為1008,a+2也就為1016。如圖6.所示
a[3][4]
a ┏━┳━┳━┳━┓
(1000)─→┃0 ┃1 ┃2 ┃3 ┃
a+1 ┣━╋━╋━╋━┫
(1008)─→┃4 ┃5 ┃6 ┃7 ┃
a+2 ┣━╋━╋━╋━┫
(1016)─→┃8 ┃9 ┃10┃11┃
┗━┻━┻━┻━┛
圖6.
既然我們把a(bǔ)[0],a[1],a[2]看成是一維數(shù)組名,可以認(rèn)為它們分別代表它們所對(duì)應(yīng)的數(shù)組的首地址,也就是講,a[0]代表第0行中第0列元素的地址,即&a[0][0],a[1]是第1行中第0列元素的地址,即&a[1][0],根據(jù)地址運(yùn)算規(guī)則,a[0]+1即代表第0行第1列元素的地址,即&a[0][1],一般而言,a[i]+j即代表第i行第j列元素的地址,即&a[i][j]。
另外,在二維數(shù)組中,我們還可用指針的形式來(lái)表示各元素的地址。如前所述,a[0]與*(a+0)等價(jià),a[1]與*(a+1)等價(jià),因此a[i]+j就與*(a+i)+j等價(jià),它表示數(shù)組元素a[i][j]的地址。
因此,二維數(shù)組元素a[i][j]可表示成*(a[i]+j)或*(*(a+i)+j),它們都與a[i][j]等價(jià),或者還可寫(xiě)成(*(a+i))[j]。
另外,要補(bǔ)充說(shuō)明一下,如果你編寫(xiě)一個(gè)程序輸出打印a和*a,你可發(fā)現(xiàn)它們的值是相同的,這是為什么呢? 我們可這樣來(lái)理解:首先,為了說(shuō)明問(wèn)題,我們把二維數(shù)組人為地看成由三個(gè)數(shù)組元素a[0],a[1],a[2]組成,將a[0],a[1],a[2]看成是數(shù)組名它們又分別是由4個(gè)元素組成的一維數(shù)組。因此,a表示數(shù)組第0行的地址,而*a即為a[0],它是數(shù)組名,當(dāng)然還是地址,它就是數(shù)組第0行第0列元素的地址。
2.2.2 指向一個(gè)由n個(gè)元素所組成的數(shù)組指針
在Turbo C中, 可定義如下的指針變量:
int (*p)[3];
指針p為指向一個(gè)由3個(gè)元素所組成的整型數(shù)組指針。在定義中,圓括號(hào)是不能少的,否則它是指針數(shù)組,這將在后面介紹。這種數(shù)組的指針不同于前面介紹的整型指針,當(dāng)整型指針指向一個(gè)整型數(shù)組的元素時(shí),進(jìn)行指針(地址)加1運(yùn)算,表示指向數(shù)組的下一個(gè)元素, 此時(shí)地址值增加了2(因?yàn)榉糯笠蜃訛?),而如上所定義的指向一個(gè)由3個(gè)元素組成的數(shù)組指針,進(jìn)行地址加1運(yùn)算時(shí),其地址值增加
了6(放大因子為2x3=6),這種數(shù)組指針在Turbo C中用得較少,但在處理二維數(shù)組時(shí),還是很方便的。例如:
int a[3][4], (*p)[4];
p=a;
開(kāi)始時(shí)p指向二維數(shù)組第0行,當(dāng)進(jìn)行p+1運(yùn)算時(shí),根據(jù)地址運(yùn)算規(guī)則,此時(shí)放大因子為4x2=8,所以此時(shí)正好指向二維數(shù)組的第1行。和二維數(shù)組元素地址計(jì)算的規(guī)則一樣,*p+1指向a[0][1],*(p+i)+j則指向數(shù)組元素a[i][j]。
例1
int a[3] [4]={
{1,3,5,7},
{9,11,13,15},
{17,19,21,23}
};
main()
{
int i,(*b)[4];
b=a+1; /* b指向二維數(shù)組的第1行, 此時(shí)*b[0]或
**b是a[1][0] */
for(i=1;i<=4;b=b[0]+2,i++)/* 修改b的指向, 每次增加2 */
printf("%d\t",*b[0]);
printf("\n");
for (i=0; i<2; i++) {
b=a+i; /* 修改b的指向, 每次跳過(guò)二維數(shù)組的
一行 */
printf("%d\t",*(b[i]+1));
}
printf ("\n");
}
程序運(yùn)行結(jié)果如下:
9 13 17 21
3 11 19
3.字符指針
我們已經(jīng)知道,字符串常量是由雙引號(hào)括起來(lái)的字符序列,例如:
"a string"
就是一個(gè)字符串常量,該字符串中因?yàn)樽址鸻后面還有一個(gè)空格字符,所以它由8個(gè)字符序列組成。在程序中如出現(xiàn)字符串常量C編譯程序就給字符串常量按排一存貯區(qū)域,這個(gè)區(qū)域是靜態(tài)的,在整個(gè)程序運(yùn)行的過(guò)程中始終占用,平時(shí)所講的字符串常量的長(zhǎng)度是指該字符串的字符個(gè)數(shù),但在按排存貯區(qū)域時(shí),C編譯程序還自動(dòng)給該字符串序列的末尾加上一個(gè)空字符'\0',用來(lái)標(biāo)志字符串的結(jié)束,因此一個(gè)字符串常量所占的存貯區(qū)域的字節(jié)數(shù)總比它的字符個(gè)數(shù)多一個(gè)字節(jié)。
Turbo C中操作一個(gè)字符串常量的方法有:
(1).把字符串常量存放在一個(gè)字符數(shù)組之中,例如:
char s[]="a string";
數(shù)組s共有9個(gè)元素所組成,其中s[8]中的內(nèi)容是'\0'。實(shí)際上,在字符數(shù)組定義的過(guò)程中,編譯程序直接把字符串復(fù)寫(xiě)到數(shù)組中,即對(duì)數(shù)組s初始化。
(2).用字符指針指向字符串,然后通過(guò)字符指針來(lái)訪問(wèn)字符串存貯區(qū)域。當(dāng)字符串常量在表達(dá)式中出現(xiàn)時(shí),根據(jù)數(shù)組的類(lèi)型轉(zhuǎn)換規(guī)則,它被轉(zhuǎn)換成字符指針。因此,若我們定義了一字符指針cp:
char *cp;
于是可用:
cp="a string";
使cp指向字符串常量中的第0號(hào)字符a,如圖7.所示。
cp
┏━━━┓ ┏━┳━┳━┳━┳━┳━┳━┳━┳━┓
┃ ─╂─→ ┃a ┃ ┃s ┃t ┃r ┃i ┃n ┃g ┃\0 ┃
┗━━━┛ ┗━┻━┻━┻━┻━┻━┻━┻━┻━┛
圖7.
以后我們可通過(guò)cp來(lái)訪問(wèn)這一存貯區(qū)域,如*cp或cp[0]就是字符a,而cp[i]或*(cp+i)就相當(dāng)于字符串的第i號(hào)字符,但企圖通過(guò)指針來(lái)修改字符串常量的行為是沒(méi)有意義的。
相關(guān)推薦:計(jì)算機(jī)等考二級(jí)C語(yǔ)言備考:C語(yǔ)言/C++編譯過(guò)程北京 | 天津 | 上海 | 江蘇 | 山東 |
安徽 | 浙江 | 江西 | 福建 | 深圳 |
廣東 | 河北 | 湖南 | 廣西 | 河南 |
海南 | 湖北 | 四川 | 重慶 | 云南 |
貴州 | 西藏 | 新疆 | 陜西 | 山西 |
寧夏 | 甘肅 | 青海 | 遼寧 | 吉林 |
黑龍江 | 內(nèi)蒙古 |