------Java培训、Android培训、iOS培训、.Net培训、期待与您交流! -------
指针数组
因为指针是变量,因此可设想用指向同一数据类型的指针来构成一个数组, 这就是指针数组。数组中的每个元素都是指针变量,根据数组的定义,指针数组中每个元素都为指向同一数据类型的指针。指针数组的定义格式为:
定义了一个指针数组,数组中的每个元素都是指向整型量的指针,该数组由10个元素组成,即a[0],a[1],a[2], ..., a[9],它们均为指针变量。a为该指针数组名,和数组一样,a是常量,不能对它进行增量运算。a为指针数组元素a[0]的地址,a+i为a[i]的地址,*a就是a[0],*(a+i)就是a[i]。
指针数组和一般数组一样,允许指针数组在定义时初始化,但由于指针数组的每个元素是指针变量,它只能存放地址,所以对指向字符串的指针数组在说明赋初值时,是把存放字符串的首地址赋给指针数组的对应元素。
char *month_name(int n) //函数month_name(n)返回一个指向包含第n月名字
{
static char *name[]={ //定义一个指针数组
Month_name, //提示字符串
January, February, March, April, //月名字的字符串常量
May, June, July, August,
September, October, November, December
};
return((n<1||n>12)?name[0]:name[n]); //返回字符串的首地址
}
main()
{
int i;
for(i=0; i<13; i++) //依次循环输出各个月的名字
printf(%s\n, month_name(i));
}
指针函数和函数指针
一、指针函数
当一个函数声明其返回值为一个指针时,实际上就是返回一个地址给调用函数,以用于需要指针或地址的表达式中。
格式:
类型说明符 * 函数名(参数)
int *GetDate();
int * aaa(int,int);
函数返回的是一个地址值,经常使用在返回数组的某一元素地址上。
int * GetDate(int wk,int dy); //函数的申明
main()
{
int wk,dy; //定义两变量存放第几星期与星期几
do
{
printf(%d\n,*GetDate(wk,dy));
printf(Enter week(1-5)day(1-7)\n); //提示输入
scanf(%d%d,&wk,&dy);
}
while(wk<1||wk>5||dy<1||dy>7); //当输入不符合要求跳出循环
}
int * GetDate(int wk,int dy) //获取日期的子函数
{
static int calendar[5][7]= //存放日期的二维数组
{
{1,2,3,4,5,6,7},
{8,9,10,11,12,13,14},
{15,16,17,18,19,20,21},
{22,23,24,25,26,27,28},
{29,30,31,-1}
};
return &calendar[wk-1][dy-1]; //返回二位数组成员的指针
}
二、函数指针
指向函数的指针包含了函数的地址,可以通过它来调用函数。声明格式如下:
类型说明符 (*函数名)(参数) void (*fptr)();
其实这里不能称为函数名,应该叫做指针的变量名。这个特殊的指针指向一个返回整型值的函数。指针的声明笔削和它指向函数的声明保持一致。
指针名和指针运算符外面的括号改变了默认的运算符优先级。如果没有圆括号,就变成了一个返回整型指针的函数的原型声明。
把函数的地址赋值给函数指针,可以采用下面两种形式:
fptr=&Function;
fptr=Function;
取地址运算符&不是必需的,因为单单一个函数标识符就标号表示了它的地址,如果是函数调用,还必须包含一个圆括号括起来的参数表。
可以采用如下两种方式来通过指针调用函数:
x=(*fptr)();
x=fptr();
第二种格式看上去和函数调用无异。但是有些程序员倾向于使用第一种格式,因为它明确指出是通过指针而非函数名来调用函数的。
void (*funcp)(); //申明一个函数指针
void FileFunc(),EditFunc(); //申明两个子函数
main()
{
funcp=FileFunc; //使函数指针指向第一个函数
(*funcp)(); //通过指针调用第一个函数
funcp=EditFunc; //使函数指针指向第二个函数
(*funcp)(); //通过指针调用第二个函数
}
void FileFunc() // 定义第一个函数
{
printf(FileFunc\n);
}
void EditFunc() // 定义第二个函数
{
printf(EditFunc\n);
}
//程序输出结果为:
//FileFunc
//EditFunc
三、指针的指针
指针的指针看上去有些令人费解。它们的声明有两个星号。例如:
char ** cp;
指针的指针需要用到指针的地址。
char c='A';
char *p=&c;
char **cp=&p;
通过指针的指针,不仅可以访问它指向的指针,还可以访问它指向的指针所指向的数据。
char *p1=*cp;
char c1=**cp;
利用指针的指针可以允许被调用函数修改局部指针变量和处理指针数组。
void FindCredit(int **); //申明一个子函数,所传递的参数为指针的指针
main()
{
int vals[]={7,6,5,-4,3,2,1,0};
int *fp=vals;
FindCredit(&fp); //调用这个子函数,并把该指针的地址作为实参传递给函数FindCredit()
printf(%d\n,*fp);
}
void FindCredit(int ** fpp) //调用该函数
{
while(**fpp!=0)
if(**fpp<0) break;
else (*fpp)++; //对形参指针指向的指针进行自增运算
}
/*首先用一个数组的地址初始化指针fp,然后把该指针的地址作为实参传递给函数FindCredit()。FindCredit()函数通过表达式**fpp间接地得到数组中的数据。为遍历数组以找到一个负值,FindCredit()函数进行自增运算的对象是调用者的指向数组的指针,而不是它自己的指向调用者指针的指针。语句(*fpp)++就是对形参指针指向的指针进行自增运算的。但是因为*运算符高于++运算符,所以圆括号在这里是必须的,如果没有圆括号,那么++运算符将作用于二重指针fpp上
*/
指向指针数组的指针
指针的指针另一用法是处理指针数组。也有人喜欢用指针数组来代替多维数组,一个常见的用法就是处理字符串。
char *Names[]= //定义一个指针数组
{
Bill, Sam, Jim, //存放各个字符串常量
Paul, Charles,
0 //循环以次来判断是否到了数组末尾具有零值的指针常常被用做循环数组的终止符
};
main()
{
char **nm=Names; //定义一个指向指针的指针变量并赋值
while(*nm!=0) //采用空指针作为终止符
printf(%s\n,*nm++); //对nm进行自增运算使其指向数组的下一个指针
}