指向函数的指针

正文

(一) 用函数指针变量调用函数

以用指针变量指向整形变量、字符串、数组、结构体、也可以指向一个函数。一个函数在编译时被分配一个入口地址。这个入口地址就称为函数指针。可以用一个指针变量指向函数,然后通过该指针变量调用此函数。用简单的数值比较为例:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
#include <stdio.h>
#include <stdlib.h>
int main()
{
int max(int,int);
int (*p)(int,int);
int a,b,c;
p = max;
scanf("%d,%d",&a,&b);
c = (*p)(a,b);
printf("a=%d,b=%d,max=%d\n",a,b,c);
return 0;
}
int max(int x,int y)
{
int z;
if(x>y) z = x;
else z = y;
return(z);
}

main函数中的" c = max(a,b); " 包括了一次函数的调用。每一个函数都占用一段内存单元。因此,可以用一个指针变量指向一个函数,通过指针变量来访问它指向的函数。

第7行:int (*p)( int,int ); 用来定义 p 是一个指向函数的指针变量, 该函数有两个整形参数,函数值为整形。注意 *p 两侧的括号不可省略,表示p 先与 * 结合,是指针变量,然后再与后面的( ) 结合,表示此指针变量指向函数,这个函数值 (即函数的返回值) 是整形的。如果写成 int *p ( int,int ),由于( )的优先级高于 *,它就成了声明一个函数P( 这个函数的返回值是指向整形变量的指针)。

赋值语句 p = max ; 作用是将函数 max 的入口地址赋给指针变量p。和数组名代表数组首元素地址类似,函数名代表该函数的入口地址。这时 p 就是指向函数 max的指针变量,此时 pmax都指向函数开头,调用 *p 就是调用 max 函数。但是p作为指向函数的指针变量,它只能指向函数入口处而不可能指向函数中间的某一处指令处,因此不能用 *(p + 1)来表示指向下一条指令。

注意:
(1) 指向函数的指针变量的一般定义形式为:

1
数据类型 (*指针变量名)(函数参数列表)

这里数据类型就是函数返回值的类型

  1. int (* p) ( int,int ); 它只是定义一个指向函数的指针变量 p, 它不是固定指向哪一个函数的,而只是表示定义这样一个类型的变量,它是专门用来存放函数的入口地址的。在程序中把哪一函数(该函数的值应该是整形的,且有两个整形参数)的地址赋给它,他就指向哪一个函数。在一个函数中,一个函数指针变量可以先后指向同类型的不同函数。

  2. p = max;在给函数指针变量赋值时,只需给出函数名而不必给出函数参数,因为是将函数的入口地址赋给p ,而不涉及 实参和形参的结合问题,不能写成 p = max(a,b);

  3. c = (*p)(a,b) 在函数调用时,只需将( *p ) 代替函数名即可,后面实参依旧。

  4. 对于指向函数的指针变量,像 p++ ,p+n…..是无意义的。

(二) 用指向函数的指针作为函数参数

函数指针变量通常的用途之一就是把指针作为参数传递到其他函数。

数的参数可以是变量、指向变量的指针变量、数组名、指向数组的指针变量,也可以是指向函数的指针也可以作为参数,以实现函数地址的传递,这样就能够在被调用的函数中使用实参函数。

1
2
3
4
5
6
7
8
9
10
11
void sub ( int ( *x1) (int), int (*x2) (int,int) )
    {
      int a,b,i,j;
      a = (*x1)(i);      /* 调用 f1 函数 */
      b = (*x2)(i)(j);    /* 调用 f2 函数 */
    }

如果实参为两个 函数名 f1f2. 在函数首部定义x1x2为函数指针变量,x1指向的函数有一个整形形参,x2指向的函数有两个形参。ij是函数f1f2所要的参数。函数sub的形参 x1x2(指针变量)在函数 sub 未被调用时并不占用内存单元,也不指向任何函数。在sub被调用时,把实参函数 f1f2的入口地址传给形式指针变量 x1x2.

既然在 sub 函数中要调用 f1f2 函数,为什么不直接调用f1f2而要用函数指针变量呢? 确实,如果只是用到f1f2 函数,完全可以在sub函数中直接调用f1f2,而不必设指针变量 x1x2。 但是,如果在每次调用sub时,调用的函数不是固定的,下次是f3f4,再是f5f6…这时用指针变量就比较方便了。

热评文章