函数指针作为参数

52

在C语言中,是否可能将函数指针作为参数传递给另一个函数?

如果可以,那么我应该如何声明和定义一个函数来接收一个函数指针作为参数呢?


1
我建议查看世界著名的C faqs - lorenzog
5个回答

88
当然。
void f(void (*a)()) {
    a();
}

void test() {
    printf("hello world\n");
}

int main() {
     f(&test);
     return 0;
}

7
两种方式都可以使用。& 符号是可选的,调用函数指针时解引用指针也是可选的。 - Mehrdad Afshari
2
没错,不需要做任何改变。而且,参数声明中的“指针”语法也是可选的。上面的 f 可以声明为 void f(void a()) - AnT stands with Russia
5
使用typedef来定义函数指针类型可以使代码更易于阅读。 - David R Tribble
有没有一种方法可以拥有一个指向函数的指针,并预定义其中一个输入参数的值?假设我们有 void f( float * data, float funParam)。我们能否拥有一个指向该函数的指针,并具有某个特定的 funParam 值? - Royi
@Royi 不行。另请参阅 https://dev59.com/HG855IYBdhLWcg3wXC5- - Antti Haapala -- Слава Україні

33
假设你有一个函数:
int func(int a, float b);

所以指向它的指针将是:
int (*func_pointer)(int, float);

然后,你可以像这样使用它:
  func_pointer = func;
  (*func_pointer)(1, 1.0);

  /*below also works*/
  func_pointer(1, 1.0);

为了避免每次需要时都要指定完整的指针类型,你可以使用typedef来定义它:
typedef int (*FUNC_PTR)(int, float);

然后像使用任何其他类型一样使用它:
void executor(FUNC_PTR func)
{ 
   func(1, 1.0);
}

int silly_func(int a, float b)
{ 
  //do some stuff
}

main()
{
  FUNC_PTR ptr;
  ptr = silly_func;
  executor(ptr); 
  /* this should also wotk */
  executor(silly_func)
}

我建议你去看一下世界著名的C语言常见问题解答(C faqs)。

18

这是一个很好的例子:

int sum(int a, int b)
{
   return a + b;
}

int mul(int a, int b)
{
   return a * b;
}

int div(int a, int b)
{
   return a / b;
}

int mathOp(int (*OpType)(int, int), int a, int b)
{
   return OpType(a, b);
}

int main()
{

   printf("%i,%i", mathOp(sum, 10, 12), mathOp(div, 10, 2));
   return 0;
}
The output is : '22, 5'

3

如其他答案所述,您可以按照以下方式进行操作:

void qsort(void *base, size_t nmemb, size_t size,
           int (*compar)(const void *, const void *));

然而,有一种特殊情况需要声明函数指针类型的参数:如果一个参数具有函数类型,则它将被转换为指向函数类型的指针,就像在参数列表中数组被转换为指针一样,因此前者也可以写成:

void qsort(void *base, size_t nmemb, size_t size,
           int compar(const void *, const void *));

当然,这仅适用于参数,在参数列表之外int compar(const void *, const void *);将声明一个函数。

+1 是为了下面提到的提高可读性的技巧,利用隐式指针转换。 - mtraceur

2

检查 qsort()

void qsort(void *base, size_t nmemb, size_t size,
           int (*compar)(const void *, const void *));

该函数的最后一个参数是一个函数指针。当您在程序中调用qsort()时,执行会通过该指针“进入库”,然后通过它“回到您自己的代码”。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接