当指针被赋值后,其在内存的安放如a ),当数据比较后进行交换,这时,指针变量与所指向的变量的关系如b )所示,在程序的运行过程中,指针变量与所指向的变量其指向始终没变。
当指针交换指向后, p 1和p 2由原来指向的变量a和b改变为指向变量b和a,这样一来, * p 1
就表示变量b,而* p 2就表示变量a。在上述程序中,无论在何时,只要指针与所指向的变量满
足p = & a;我们
就可以对变量a 以指针的形式来表示。此时p等效于& a,* p等效于变量a 。
6.3.2 指针变量作函数的参数
函数的参数可以是我们在前面学过的简单数据类型,也可以是指针类型。使用指针类型做函数的参数,实际向函数传递的是变量的地址。由于子程序中获得了所传递变量的地址,在该地址空间的数据当子程序调用结束后被物理地保留下来。
[例6-4] 利用指针变量作为函数的参数,用子程序的方法再次实现上述功能。
main( )
{
void chang(); / *函数声明* /
int *p1,*p2,a,b,*t;
scanf("%d, %d", &a, &b);
p1 = &a;
p2 = &b;
chang(p1 , p2); / *子程序调用* /
printf("%d, %d\n" , *p1, *p2);
return 0;
}
void chang(int *pt1,int *pt2)
{ / *子程序实现将两数值调整为由大到小* /
int t;
if (*pt1<*pt2) / *交换内存变量的值* /
{
t=*pt1; *pt1=*pt2; * pt2 = t;
}
return;
}
由于在调用子程序时,实际参数是指针变量,形式参数也是指针变量,实参与形参相结合,传值调用将指针变量传递给形式参数pt1和pt2。但此时传值传递的是变量地址,使得在子程序中pt1和pt2具有了p1和p2的值,指向了与调用程序相同的内存变量,并对其在内存存放的数据进行了交换,其效果与[例6 - 2 ]相同。
思考下面的程序,是否也能达到相同的效果呢?
main( )
{
void chang();
int *p1,*p2,a,b,*t;
scanf("%d,%d", &a, &b);
p1 = &a;
p2 = &b;
chang(p1, p2);
printf("%d, %d\n" , *p1, *p2);
}
void chang(int *pt1,int *pt2)
{
int *t;
if (*pt1<*pt2)
{
t=pt1; pt1=pt2; p t 2 = t ;
}
return;
}
程序运行结束,并未达到预期的结果,输出与输入完全相同。其原因是对子程序来说,函数内部进行指针相互交换指向,而在内存存放的数据并未移动,子程序调用结束后,main( )函数中p 1和p 2保持原指向,结果与输入相同。
6.4 指针与数组 变量在内存存放是有地址的,数组在内存存放也同样具有地址。对数组来说,数组名就是数组在内存安放的首地址。指针变量是用于存放变量的地址,可以指向变量,当然也可存放数组的首址或数组元素的地址,这就是说,指针变量可以指向数组或数组元素,对数组而言,数组和数组元素的引用,也同样可以使用指针变量。下面就分别介绍指针与不同类型的数组。
6.4.1 指针与一维数组 假设我们定义一个一维数组,该数组在内存会有系统分配的一个存储空间,其数组的名字就是数组在内存的首地址。若再定义一个指针变量,并将数组的首址传给指针变量,则该指针就指向了这个一维数组。我们说数组名是数组的首地址,也就是数组的指针。而定义的指针变量就是指向该数组的指针变量。对一维数组的引用,既可以用传统的数组元素的下标法,也可使用指针的表示方法。
int a[10] , *ptr; /* 定义数组与指针变量* /
做赋值操作:ptr=a; 或ptr = &a[0];
则ptr就得到了数组的首址。其中, a是数组的首地址, &a[0]是数组元素a[0]的地址,由于a[0]的地址就是数组的首地址,所以,两条赋值操作效果完全相同。指针变量ptr就是指向数组a的指针变量。
若ptr指向了一维数组,现在看一下C规定指针对数组的表示方法:
1) ptr+n与a + n表示数组元素a[n]的地址,即&a[n] 。对整个a数组来说,共有10个元素, n的取值为0~9,则数组元素的地址就可以表示为ptr + 0~ptr + 9或a + 0~a + 9,与&a[0] ~&a[9]保持一致。
2) 知道了数组元素的地址表示方法, *(ptr + n)和*(a+n)就表示为数组的各元素即等效于a[n]。
3) 指向数组的指针变量也可用数组的下标形式表示为ptr[n],其效果相当于*(ptr + n)。
[例6-5] /*以下标法输入输出数组各元素。
下面从键盘输入10个数,以数组的不同引用形式输出数组各元素的值。
# include <stdio.h>
main( )
{
int n,a[10],*ptr=a;
for(n = 0; n<=9; n++)
scanf("%d" , &a[n]);
printf("1------output! \n");
for(n = 0; n<=9; n++)
printf("%4d", a[n]);
printf("\n");
}
运行程序:
1 2 3 4 5 6 7 8 9 0
1------output!
1 2 3 4 5 6 7 8 9 0
[例6-6] 采用指针变量表示的地址法输入输出数组各元素。
#include<stdio.h>
main( )
{
int n,a[10],*ptr=a; / *定义时对指针变量初始化* /
for(n = 0; n<=9; n++)
scanf("%d", ptr + n);
printf("2------output! \n");
for(n=0; n<=9; n++)
printf("%4d", *(ptr+n));
printf("\n");
}
运行程序:
1 2 3 4 5 6 7 8 9 0
2------output!
1 2 3 4 5 6 7 8 9 0
[例6-7] 采用数组名表示的地址法输入输出数组各元素。
main( )
{
int n,a[10],*ptr=a;
for(n = 0; n < = 9; n ++)
scanf("%d", a+n);
printf("3------output! \n");
for(n = 0; n<=9; n++)
printf("%4d", *(a+n));
printf("\n");
}
运行程序:
1 2 3 4 5 6 7 8 9 0
3------output!
1 2 3 4 5 6 7 8 9 0
[例6-8] 用指针表示的下标法输入输出数组各元素。
main( )
{
int n,a[10],*ptr=a;
for(n = 0; n<=9; n++)
scanf("%d", &ptr[n]);
printf("4------output! \n");
for(n = 0; n<=9; n++)
printf("%4d", ptr[n]);
printf("\n");
}
运行程序:
1 2 3 4 5 6 7 8 9 0
4----output!
1 2 3 4 5 6 7 8 9 0
[例6-9] 利用指针法输入输出数组各元素
main( )
{
int n,a[10],*ptr=a;
for(n = 0; n<=9; n++)
scanf("%d", ptr++);
printf("5------output! \n");
ptr = a; / *指针变量重新指向数组首址* /
for(n = 0; n<=9; n++)
printf("%4d", *ptr++);
printf("\n");
}
运行程序:
1 2 3 4 5 6 7 8 9 0
5-----output!
1 2 3 4 5 6 7 8 9 0
在程序中要注重*ptr++所表示的含义。*ptr表示指针所指向的变量; ptr++表示指针所指向的变量地址加1个变量所占字节数,具体地说,若指向整型变量,则指针值加2,若指向实型,则加4,依此类推。而printf(“%4d”, *ptr+ +)中,*ptr++所起作用为先输出指针指向的变量的值,然后指针变量加1。循环结束后,指针变量指向如图6 - 6所示:
指针变量的值在循环结束后,指向数组的尾部的后面。假设元素a[9]的地址为1000,整型占2字节,则ptr的值就为100 2。请思考下面的程序段:
main( )
{
int n,a[10],*ptr=a;
for(n = 0; n<=9; n++)
scanf("%d", ptr++);
printf("4------output! \n");
for(n = 0; n<=9; n++)
printf("%4d", *ptr++);
printf("\n");
}
程序与例6 - 9相比,只少了赋值语句p t r = a;程序的运行结果还相同吗?
6.4.2 指针与二维数组 定义一个二维数组:
int a[3][4];
表示二维数组有三行四列共1 2个元素,在内存中按行存放,存放形式为图6 - 7:
其中a是二维数组的首地址, &a[0][0]既可以看作数组0行0列的首地址,同样还可以看作是二维数组的首地址, a [0]是第0行的首地址,当然也是数组的首地址。同理a[n]就是第n行的首址;&a[n][m]就是数组元素a[n][m]的地址。
既然二维数组每行的首地址都可以用a[n]来表示,我们就可以把二维数组看成是由n行一维数组构成,将每行的首地址传递给指针变量,行中的其余元素均可以由指针来表示。图6 - 8给出了指针与二维数组的关系。
我们定义的二维数组其元素类型为整型,每个元素在内存占两个字节,若假定二维数组从1000单元开始存放,则以按行存放的原则,数组元素在内存的存放地址为1000~1022。
用地址法来表示数组各元素的地址。对元素a[1][2],&a[1][2]是其地址, a[1] + 2也是其地址。分析a[1] + 1与a [1] + 2的地址关系,它们地址的差并非整数1,而是一个数组元素的所占位置2,原因是每个数组元素占两个字节。
对0行首地址与1行首地址a与a + 1来说,地址的差同样也并非整数1,是一行,四个元素占的字节数8。由于数组元素在内存的连续存放。给指向整型变量的指针传递数组的首地址,则该指针指向二维数组。
int *ptr, a[3][4];
若赋值: p t r = a;则用ptr++ 就能访问数组的各元素。
[例6-10] 用地址法输入输出二维数组各元素。
#include <stdio.h>
main( )
{
int a[3][4];
int i,j;
for(i = 0; i<3; i++)
for(j = 0; j<4; j++)
scanf("%d",a[i]+j); / *地址法* /
for(i = 0; i<3; i++)
{
for(j = 0; j<4; j++)
printf("%4d",*(a[i]+j)); /* *(a[i]+j) 是地址法所表示的数组元素* /
printf("\n");
}
}
运行程序:
1 2 3 4 5 6 7 8 9 10 11 12
1 2 3 4
5 6 7 8
9 1 0 1 1 1 2
[例6 - 11] 用指针法输入输出二维数组各元素。
#include<stdio.h>
main( )
{
int a[3][4],*ptr;
int i,j;
ptr = a[0];
for(i = 0; i<3; i++)
for(j = 0; j< 4; j++)
scanf("%d", ptr++); / *指针的表示方法* /
ptr = a[0];
for(i = 0; i<3; i++)
{
for(j = 0; j<4; j++)
printf("%4d", *ptr++);
printf("\n");
}
}
运行程序:
1 2 3 4 5 6 7 8 9 10 11 12
1 2 3 4
5 6 7 8
9 1 0 1 1 1 2
对指针法而言,程序可以把二维数组看作展开的一维数组:
main( )
{
int a[3][4],*ptr;
int i,j;
ptr = a[0];
for(i = 0; i< 3; i++)
for(j = 0; j < 4; j++)
scanf("%d", ptr++); / * 指针的表示方法* /
ptr = a[0];
for(i = 0; i < 12;i++)
printf("%4d", *ptr++);
printf("\n");
}
运行程序:
1 2 3 4 5 6 7 8 9 10 11 12
1 2 3 4 5 6 7 8 9 1 0 1 1 1 2
6.4.3 数组指针作函数的参数
学习了指向一维和二维数组指针变量的定义和正确引用后,我们现在学习用指针变量作函数的参数。
[例6-12] 调用子程序,实现求解一维数组中的最大元素。
我们首先假设一维数组中下标为0的元素是最大和用指针变量指向该元素。后续元素与该元素一一比较,若找到更大的元素,就替换。子程序的形式参数为一维数组,实际参数是指向一维数组的指针。
#include <stdio.h>
main( )
{
int sub_max(); / * 函数声明* /
int n,a[10],*ptr=a; / *定义变量,并使指针指向数组* /
int max;
for(n = 0; n < = i - 1; n++) / *输入数据* /
scanf("%d", &a[n]);
max = sub_max(ptr, 10); / * 函数调用,其实参是指针* /
printf("max = %d\n", max);
}
int sub_max(b,i) / * 函数定义,其形参为数组* /
int b[],i;
{
int temp,j;
temp = b[0];
for(j = 1; j < = 9; j++)
if(temp<b[j]) temp=b[j];
return temp;
}
程序的main( )函数部分,定义数组a 共有1 0个元素,由于将其首地址传给了ptr,则指针变量ptr 就指向了数组,调用子程序,再将此地址传递给子程序的形式参数b,这样一来,b数组在内存与a 数组具有相同地址,即在内存完全重合。在子程序中对数组b 的操作,与操作数组a 意义相同。其内存中虚实结合的示意如图6 - 9所示。
main( )函数完成数据的输入,调用子程序并输出运行结果。sub_max( )函数完成对数组元素找最大的过程。在子程序内数组元素的表示采用下标法。运行程序:
1 3 5 7 9 2 4 6 8 0
max = 9
视频教程列表
文章教程搜索
C语言程序设计推荐教程
C语言程序设计热门教程