一个指针变量可以指向整型变量、实型变量、字符类型变量,当然也可以指向指针类型变量。当这种指针变量用于指向指针类型变量时,我们称之为指向指针的指针变量,这话可能会感到有些绕口,但你想到一个指针变量的地址就是指向该变量的指针时;这种双重指针的含义就容易理解了。下面用一些图来描述这种双重指针,见图6-13。 在图中,整型变量i的地址是&i,将其传递给指针变量p,则p指向i;实型变量j的地址是&j,将其传递给指针变量p,则p指向j;字符型变量ch的地址是&ch,将其传递给指针变量p,则p指向ch;整型变量x的地址是&x,将其传递给指针变量p2,则p2指向x,p2是指针变量,同 时,将p2的地址&p2传递给p1,则p1指向p2。这里的p1就是我们谈到的指向指针变量的指针变量,即指针的指针。 指向指针的指针变量定义如下: 类型标识符**指针变量名 例如:float**ptr; 其含义为定义一个指针变量ptr,它指向另一个指针变量(该指针变量又指向一个实型变量)。由于指针运算符“*”是自右至左结合,所以上述定义相当于: float*(*ptr); 下面看一下指向指针变量的指针变量怎样正确引用。 [例6-27]用指向指针的指针变量访问一维和二维数组。 #include<stdio.h> #include<stdlib.h> main() { int a[10],b[3][4],*p1,*p2,**p3,i,j;/是*p指3向指针的指针变量*/ for(i=0;i<10;i++) scanf("%d",&a[i]);/*一维数组的输入*/ for(i=0;i<3;i++) for(j=0;j<4;j++) scanf("%d",&b[i][j]);/*二维数组输入*/ for(p1=a,p3=&p1,i=0;i<10;i++) printf("%4d",*(*p3+i));/*用指向指针的指针变量输出一维数组*/ printf("/n"); for(p1=a;p1-a<10;p1++)/*用指向指针的指针变量输出一维数组*/ { p3=&p1; printf("%4d",**p3); } printf("/n"); for(i=0;i<3;i++)/*用指向指针的指针变量输出二维数组*/ { p2=b[i]; p3=&p2; for(j=0;j<4;j++) printf("%4d",*(*p3+j)); printf("/n"); } for(i=0;i<3;i++)/*用指向指针的指针变量输出二维数组*/ { p2=b[i]; for(p2=b[i];p2-b[i]<4;p2++) { p3=&p2; printf("%4d",**p3); } printf("/n"); } } 程序的存储示意如图6-14所示,对一维数组a来说,若把数组的首地址即数组名赋给指针变量p1,p1就指向数组a,数组的各元素用p1表示为,*(p1+i),也可以简化为*p1+i表示。 如果继续作将p3=&p1,则将p1的地址传递给指针变量p3,*p3就是p1。用p3来表示一维数组的各元素,只需要将用p1表示的数组元素*(p1+i)中的p1换成*p3即可,表示为*(*p3+i)。 同样,对二维数组b来说,b[i]表示第i行首地址,将其传递给指针变量p2,使其指向该行。 该行的元素用p2表示为*(p2+i)。若作p3=&p2,则表示p3指向p2,用p3表示的二维数组第i行元素为:*(*p3+i)。这与程序中的表示完全相同。 运行程序: [例6-28]利用指向指针的指针变量对二维字符数组的访问。 #include<stdio.h> #include<stdlib.h> main() { int i; staticcharc[][16]={"clanguage","fox","computer","homepage"}; /*二维字符数组*/ static char *cp[]={c[0],c[1],c[2],c[3]};指/*针数组*/ static char **cpp;/*指向字符指针的指针变量*/ cpp=cp;/*将指针数组的首地址传递给指向字符指针的指针变量*/ for(i=0;i<4;i++)/*按行输出字符串*/ printf("%s/n",*cpp++); printf("-----------/n"); for(i=0;i<4;i++)/*按行输出字符串*/ { cpp=&cp[i]; printf("%s/n",*cpp); } } 程序中需要注意的是,执行cpp=cp是将指针数组的首地址传递给双重指针,所以*(cpp+i)表示第i行的首地址,而不是cpp+i。在程序设计时一定分清。 |