一个指针变量可以指向整型变量、实型变量、字符类型变量,当然也可以指向指针类型变量。当这种指针变量用于指向指针类型变量时,我们称之为指向指针的指针变量,这话可能会感到有些绕口,但你想到一个指针变量的地址就是指向该变量的指针时;这种双重指针的含义就轻易理解了。下面用一些图来描述这种双重指针,见图6 - 1 3。
在图中,整型变量i的地址是& i,将其传递给指针变量p,则p指向i;实型变量j的地址是& j,将其传递给指针变量p,则p指向j; 字符型变量c h的地址是&ch,将其传递给指针变量p,则p指向ch; 整型变量x的地址是&x,将其传递给指针变量p2,则p2指向x,p2是指针变量,同时,将p2的地址&p2传递给p1,则p1指向p2。这里的p 1就是我们谈到的指向指针变量的指针变量,即指针的指针。
指向指针的指针变量定义如下:
类型标识符* *指针变量名
例如: 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; /*p3 是指向指针的指针变量* /
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 - 1 4所示,对一维数组a来说,若把数组的首地址即数组名赋给指针变量p1,p1就指向数组a,数组的各元素用p1表示为, *(p1 + i),也可以简化为*p1 + i表示。
假如继续作将p3 = &p1,则将p1的地址传递给指针变量p3,*p3就是p1。用p3来表示一维数组的各元素,只需要将用p 1表示的数组元素*(p1 + i)中的p1换成*p3即可,表示为*(*p3 + i)。
图6-14 例6-27程序的存储示意图
同样,对二维数组b来说,b[i]表示第i行首地址,将其传递给指针变量p2,使其指向该行。该行的元素用p 2表示为*(p2+i)。若作p3 = &p2,则表示p3指向p2,用p3表示的二维数组第i行元素为:*(*p3 + i)。这与程序中的表示完全相同。
运行程序:
1 2 3 4 5 6 7 8 9 0
1 3 5 7
2 4 6 8
5 7 9 2
1 2 3 4 5 6 7 8 9 0
1 2 3 4 5 6 7 8 9 0
1 3 5 7
2 4 6 8
5 7 9 2
1 3 5 7
2 4 6 8
5 7 9 2
[例6-28] 利用指向指针的指针变量对二维字符数组的访问。
#include<stdio.h>
#include<stdlib.h>
main( )
{
int i;
static char c[][16]={"c language","fox","computer","home page"};
/ *二维字符数组* /
static char *cp[]={c[0],c[1],c[2],c[3]};/* 指针数组* /
static char **cpp; /* 指向字符指针的指针变量* /
c p p = c p ; / *将指针数组的首地址传递给指向字符指针的指针变量* /
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);
}
}
运行程序:
c language
fox
computer
home page
----------
c language
fox
computer
home page
程序中需要注重的是,执行cpp = cp是将指针数组的首地址传递给双重指针,所以*(cpp + i)表示第i行的首地址,而不是cpp + i。在程序设计时一定分清。
6.8 main函数的参数 C程序最大的特点就是所有的程序都是用函数来装配的。main( )称之为主函数,是所有程序运行的入口。其余函数分为有参或无参两种,均由main( )函数或其它一般函数调用,若调用的是有参函数,则参数在调用时传递。
main( )
{
. . .
y1 = f1(x1, x2);
. . .
}
f1(int a,int b)
{
. . . .
Y 2 = f 2 ( x 3 , x 4 ) ;
. . . .
}
f2( int m,int n)
{
. . . .
. . . . .
}
在前面课程的学习中,对main( )函数始终作为主调函数处理,也就是说,答应main( )调用其它函数并传递参数。事实上, main( )函数既可以是无参函数,也可以是有参的函数。对于有参的形式来说,就需要向其传递参数。但是其它任何函数均不能调用main( )函数。当然也同样无法向main( )函数传递,只能由程序之外传递而来。这个具体的问题怎样解决呢?
我们先看一下main( )函数的带参的形式:
main(argc, argv)
int argc,char * argv[];
{
. . . . .
}
从函数参数的形式上看,包含一个整型和一个指针数组。当一个C的源程序经过编译、链接后,会生成扩展名为. E X E的可执行文件,这是可以在操作系统下直接运行的文件,换句话说,就是由系统来启动运行的。对main( )函数既然不能由其它函数调用和传递参数,就只能由系统在启动运行时传递参数了。
在操作系统环境下,一条完整的运行命令应包括两部分:命令与相应的参数。其格式为:
命令参数1 参数2 . . . .参数n¿
此格式也称为命令行。命令行中的命令就是可执行文件的文件名,其后所跟参数需用空格分隔,并为对命令的进一步补充,也即是传递给main( )函数的参数。
命令行与main( )函数的参数存在如下的关系:
设命令行为: program str1 str2 str3 str4 str5
其中program 为文件名, 也就是一个由program.c 经编译、链接后生成的可执行文件program.exe,其后各跟5个参数。对main( )函数来说,它的参数arc记录了命令行中命令与参数的个数,共6个,指针数组的大小由参数argc的值决
定,即为char *argv[6],指针数组的取值情况如图6 - 1 5所示。
数组的各指针分别指向一个字符串。应当引起注重的是接收到的指针数组的各指针是从命令行的开始接收的,首先接收到的是命令,其后才是参数。
下面用实例来说明带参数的main( )函数的正确使用。
[例6-29] 利用图形库函数绘制一个变化的环。它是把一个半径为R 1的圆周分成n份,然后以每个等分点为圆心,以R s为半径画n个圆(关于作图的具体理论本教材第9章第1节作了专门介绍,这里只作简单分析)。利用main( )函数的带参数形式,我们可以从键盘以命令行的方式输入R1和Rs及屏幕的背景色。
#include<graphics.h> /*包含图形库函数的头文件* /
#include<math.h>
#define pi 4.1415926
main(argc, argv)
int argc;char *argv[]; /* 定义带参数的main( ) * /
{
int x,y,r1,rs,color;
double a;
int gdriver=DETECT,gmode;
initgraph(&gdriver,&gmode,"..\\bgi ");/* 启动图形工作方式* /
r1 = atoi(argv[1]); / *计算基础圆半径* /
rs = atoi(argv[2]); / *计算同心圆半径* /
color = atoi(argv[3]); / *背景色* /
cleardevice( ); / *清除图形屏幕* /
setbkcolor(color); / *设置背景色* /
setcolor(4); / *设置图形显示颜色* /
for(a=0; a<=2*pi;a+=pi/18) / *绘制同心圆* /
{
x = r1 * cos(a) + 320;
y = r1 * sin(a) + 240;
circle(x, y, rs); / *以圆心坐标为x、y,半径为r s画圆* /
}
getch( ) ; / *等待按键继续* /
closegraph( ) ; / *关闭图形工作方式* /
}
若程序名为L 6 - 29.c,经编译、连结生成可执行文件L6 - 29.exe。在操作系统的环境下运行程序,命令行方式为:
l6-29 40 20 3
则命令行与main( )函数的参数有如图6 - 16所示的关系。
图6 - 16中, argv[0]是程序名, argv[1]是r1的值,argv[2]是rs的值,argv[3]是屏幕的背景色。由于指针数组均存放字符串,所需的圆半径及背景色彩通过atoi( )函数转换为整型。
通过带参数的main( )函数,我们可以为自己的程序设置口令,在运行程序的命令行中给出所需的口令,正确则继续,否则退出。程序图形输出如图6 - 17所示。
[例6-30] 将上述程序作修改,在程序的入口处添置密码,若给定密码正确,则显示图形。
#include<graphics.h>
#include<math.h>
#define pi 4.1415926
main(argc, argv)
int argc;char *argv[];
{
int x,y,r1,rs,color;
double a;
int gdriver=DETECT,gmode;
if(strcmp(argv[1],"pass")!=0) /* 设置口令的比较* /
{
printf("password error!\n");
exit(0);
}
initgraph(&gdriver,&gmode,"..\\bgi ");
r1 = atoi(argv[2]);
rs = atoi(argv[3]);
color = atoi(argv[4]);
cleardevice( );
setbkcolor(color);
setcolor(4);
for(a=0; a<=2*pi;a+=pi/18)
{
x = r1 * cos(a) + 320;
y = r1 * sin(a) + 240;
circle(x, y, rs);
}
getch( );
closegraph( );
}
在操作系统的环境下运行程序, 命令行中增加口令“pass”命令行方
式为:
l6-30 pass 20 40 3
指针数组的存储字符串如图6 - 18所示。
若给定字符串argv[1]的值是pass,则程序正确运行,否则程序退出。口令正确的情况下,显示的图形为图6 - 17中的一个。