函数指针的用法

函数指针的用法

 

            By zieckey

       在程序运行中,函数代码是程序的算法指令部分,它们和数组一样也占用存储空间,都有相应的地址。可以使用指针变量指向数组的首地址来调用数组数据,也可以使用指针变量指向函数代码的首地址来调用函数,指向函数代码首地址的指针变量就称为函数指针。

1.函数指针定义

函数类型 (*指针变量名)(形参列表);

“函数类型”说明函数的返回类型,由于“()”的优先级高于“*”,所以指针变量名外的括号必不可少,后面的“形参列表”表示指针变量指向的函数所带的参数列表。

例如:

int (*f)(int x);

double (*ptr)(double x);

在定义函数指针时请注意:
   
函数指针和它指向的函数的参数个数和类型都应该是—致的;

函数指针的类型和函数的返回值类型也必须是一致的。

2.函数指针的赋值

函数名和数组名一样代表了函数代码的首地址,因此在赋值时,直接将函数指针指向函数名就行了。

例如,

int func(int x);   /* 声明一个函数 */

int (*f) (int x);    /* 声明一个函数指针 */

 把函数的地址赋值给函数指针,可以采用下面两种形式:
      
f=&func;            /* 将func函数的首地址赋给指针f */   
f=func;

赋值时函数func不带括号,也不带参数,由于func代表函数的首地址,因此经过赋值以后,指针f就指向函数func(x)的代码的首地址。
第二种格式看上去和函数调用无异。但是有些程序员倾向于使用第一种格式,因为它明确指出是通过指针而非函数名来调用函数的。

3、下面给出一个示例程序:


#include <stdio.h>
void (*fp)(); //函数指针: 指向函数的指针包含了函数的地址,可以通过它来调用函数

void (*fp_para)(char *);//带参数的函数指针


void func_para(char *s);
void func();

int main(void)
{
    fp_para=&func_para;//或者:fp_para=func_para;

    (*fp_para)("This is the parameter of the funtion!");

    fp=&func;//或者:fp=&func;

    (*fp)();

    return 0;
}

void func_para(char *s)
{
    printf("%s/n", s);
}

void func()
{
    printf("This function has no parameter!/n");
}

<think>好的,用户想了解C或C++中函数指针的使用方法和示例。首先,我需要回忆一下函数指针的基本概念。根据引用[2]和[3],函数指针是指向函数的指针变量,可以用来调用函数。那用户可能需要知道如何声明、初始化和使用函数指针。接下来,我得整理一下结构。先从定义开始,说明函数指针的声明方式,比如如何根据函数原型来声明指针。例如,一个返回int,参数为两个int的函数,其指针类型是int(*)(int,int)。然后,给出赋值和调用的方法,比如直接指向函数名或者使用取地址符号,调用时可以用(*ptr)()或者直接ptr()。 然后,考虑应用场景,比如回调函数,这在排序或事件处理中很常见。引用[3]中的例子是Add函数,可以演示如何用函数指针调用。另外,引用[2]中的例子展示了指针函数和函数指针的区别,可能需要强调两者的不同,避免混淆。 还要注意示例代码的正确性,确保语法正确,比如在C++中使用using或typedef来简化声明。比如typedefint (*FuncPtr)(int, int)可以让代码更易读。此外,动态绑定和多态的例子,比如根据输入选择不同的操作函数,这能展示函数指针的灵活性。用户可能还关心常见错误,比如函数指针类型不匹配导致的错误,或者空指针调用的问题。但用户的问题主要是用法和示例,这部分可能需要放在注意事项里简要提及。最后,生成相关问题,帮助用户深入理解,比如函数指针指针函数的区别,回调函数的应用,C++中的改进方式如std::function等。这样用户可以有进一步学习的路径。 需要确保回答结构清晰,分点说明,每个部分有示例代码,并正确引用提供的资料,比如引用[2]和[3]中的代码片段。同时,检查LaTeX格式是否符合要求,比如行内公式用$...$,独立公式用$$...$$,不过这个问题可能不涉及数学公式,但代码块需要用正确标记。</think>在C/C++中,函数指针是实现动态行为调用的重要工具,以下从5个核心场景详解其用法: ### 一、基础定义与调用 函数指针的声明遵循函数原型规则: ```c int (*func_ptr)(int, int); // 声明指向函数的指针 ``` 赋值时可直接取函数地址: ```c int Add(int a, int b) { return a + b; } func_ptr = &Add; // 或 func_ptr = Add; ``` 调用方式具有等价性: ```c int result = (*func_ptr)(3,5); // 传统方式 int result = func_ptr(3,5); // C++简化写法 ``` ### 二、回调函数实战 在排序算法中应用(参考qsort原型): ```c void qsort(void *base, size_t nmemb, size_t size, int (*compar)(const void *, const void *)); int compare_int(const void *a, const void *b) { return *(int*)a - *(int*)b; } int arr[] = {5,2,8,1}; qsort(arr, 4, sizeof(int), compare_int); // 通过函数指针注入比较逻辑[^2] ``` ### 三、结构体封装 创建可扩展的接口: ```c typedef struct { int (*open)(const char*); void (*close)(int); } DeviceDriver; int usb_open(const char *dev) { /*...*/ } void usb_close(int fd) { /*...*/ } DeviceDriver usb_driver = {usb_open, usb_close}; ``` ### 四、多态实现 运行时动态绑定: ```c++ void encrypt_AES(const char* data) { /*...*/ } void encrypt_RSA(const char* data) { /*...*/ } void process_data(const char* data, void (*encryptor)(const char*)) { encryptor(data); // 动态选择加密算法 } ``` ### 五、类型安全进阶(C++11) 使用`using`提升可读性: ```cpp using MathFunc = int(*)(int,int); // C++11类型别名 MathFunc func = std::max; ``` #### 注意事项 1. 匹配返回值和参数类型 2. 空指针检查:`if(func_ptr != nullptr)` 3. C++推荐使用`std::function`和lambda表达式 ```mermaid graph TD A[函数指针声明] --> B[赋值函数地址] B --> C[调用函数指针] C --> D[回调函数应用] D --> E[多态实现] E --> F[类型安全封装] ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值