一个函数不仅可以返回一个整形数据的值、字符类型值和实型类型的值,还可以返回指针类型的数据,使其指向某个地址单元。指针函数即返回指针的函数。
指针函数一般定义格式为:
类型标识符 *函数名(参数表);
其中,后缀运算符括号()表示这是一个函数,其前缀运算符星号*表示此函数为指针型函数,其函数值为指针,即它带回来的值的类型为指针,当调用这个函数后,将得到一个指向返回值为…的指针(地址)。
比如:int *f(int x, int y);
由于*的优先级低于()的优先级,因而f首先和后面的()结合,也就意味着,f是一个函数。即:
int *(f(int x, int y) ;
接着再和前面的*结合,说明这个函数的返回值是一个指针。由于前面还有一个int,也就是说,f是一个返回值为整型指针的函数。
其中x,y是形式参数,f是函数名,调用后返回一个指向整型数据的地址指针,f(int x, int y)是函数,其返回值是指针。
//test.cpp
#include <iostream>
using namespace std;
char *display(char *str)
{
return str;
}
int main()
{
char *str = "hello";
char *out = display(str);
cout << out << endl;
return 0;
}
程序运行截图:
二 函数指针
(1)函数指针的定义
在程序运行中,函数代码是程序的算法指令部分,它们和数组一样也占用存储空间,都有相应的地址,可以使用指针变量指向数组的首地址,也可以使用指针变量指向函数的首地址,指向函数代码首地址的指针变量称为函数指针。
函数指针一般定义格式:
类型标识符 (*指针变量名)(形参列表);
由于()的优先级高于*,所以指针变量名外的括号是必不可少的。
比如:int (*f)(int x);
通过括号强行将f首先与“*”结合,也就意味着,f是一个指针,接着与后面的“()”结合,说明该指针指向的是一个函数,然后再与前面的int结合,也就是说,该函数的返回值是int。由此可见,f是一个指向返回值为int的函数的指针。
定义函数指针时应该注意:
1 函数指针和它指向的函数的参数个数和类型都应该是一致的
2 函数指针的类型和函数的返回值类型也必须是一致的
(2)函数指针的赋值
函数名和数组名一样代表了函数代码的首地址,因此在赋值时,可以直接将函数指针指向函数名
比如:
int func(int x); //声明一个函数
int (*f)(int x); //声明一个函数指针
f = func; //将func函数的首地址赋值给函数指针f
赋值时函数func不带括号,也不带参数,由于func代表函数的首地址,因此经过赋值以后,函数指针f就指向函数func(x)的代码的首地址
(3)通过函数指针调用函数
函数指针是通过函数名及有关参数来进行调用的,与其他指针变量类似,如果指针变量pi是指向某整型变量i的指针,则*pi等于它所指向的变量i。同样的,f是指向函数func(x)的指针,则*f就代表它所指向的函数func。所以在执行了f=func之后,*f和func代表同一函数。
由于函数指针指向存储区中得某个函数,因此可以通过函数指针调用相应的函数,该过程分为以下三步:
1 声明一个函数指针变量
比如:int (*f)(int x);
2 对函数指针变量赋值
比如:f=func; (func必须先要有定义)
3 要用(*指针变量)(参数表)的格式来调用函数
比如:(*f)(x) (x必须先赋值)
//test.cpp
#include <iostream>
using namespace std;
int max(int x, int y)
{
return x > y ? x : y;
}
//函数指针
int (*pMax)(int x, int y);
int main()
{
int x = 5;
int y = 10;
//给函数指针赋值
pMax = max;
int result = (*pMax)(x, y);
cout << "The Max Is: " << result << endl;
return 0;
}
程序运行截图: