自学内容网 自学内容网

C语言之函数指针(持续更新)

C语言精髓是指针,指针知识深似海,遇到一些学习一些~


1. typedef 定义函数指针类型

函数参数化是指通过函数指针将函数的某些行为参数化。这样,我们可以在调用函数时动态地指定函数的行为。以下是一个示例:

#include <stdio.h>

void process_array(int *array, size_t size, int (*process)(int))
{
    for(size_t i = 0; i < size; i++)
    {
        array[i] = process(array[i]);
    }
}

int increment(int n)
{
    return n * 2;
}


int main()
{
    int array[] = {1, 2, 3, 4, 5};
    size_t size = sizeof(array) / sizeof(int);
    process_array(array, size, increment);
    for(size_t i =0; i < size; i++)
    {
        printf("%d", array[i]);
    }
    printf("\n");
    return 0;
}

如果通过typedef来定义函数指针类型:

#include <stdio.h>

typedef int (*process_t)(int);

void process_array(int *array, size_t size, process_t process)
{
    for(size_t i = 0; i < size; i++)
    {
        array[i] = process(array[i]);
    }
}

int increment(int n)
{
    return n * 2;
}


int main()
{
    int array[] = {1, 2, 3, 4, 5};
    size_t size = sizeof(array) / sizeof(int);
    process_array(array, size, increment);
    for(size_t i =0; i < size; i++)
    {
        printf("%d", array[i]);
    }
    printf("\n");
    return 0;
}

定义的 process_array 函数接受三个参数:一个整型数组、数组大小、一个函数指针。函数指针指向一个函数,该函数接受一个整型参数并返回一个整型结果因此,当执行到 array[i] = process(array[i]); 时,process函数会自动调用increment函数。
有人可能会问了:“那如果我再定义一个函数:int aaaa(int x){return x + 1;}呢?process 函数会调用哪个函数?”。
这样符合 process_t 类型的函数就有多个了,我们可以将 process 指针指向某个函数,比如:

process_t process = aaaa; // 设置 process 指针指向 aaaa 函数
process_array(array, size, process);

2. void* 空指针的解引用

首先定义一个uint32_t 类型的param_value ,其值为0xa5a5,现在已经通过某种方法将 param_value 的地址指针传入了函数 sys_manager_handler 中(第三个参数),我们需要关注的是如何在函数内使用printf将 param_value 这个值打印出来。
在printf中,应该使用何种占位符?param作为一个指针又应该如何变换?
伪代码如下:

uint32_t param_value = 0xa5a5;

int sys_manager_handler(void *cobj, uint32_t event_id, void *param)
{
    printf("we got value from event:%lx", *(uint32_t *)param);
    printf("sys_manger got value from event:%lx", *param);
}

以上两句 printf 哪个正确?
由于我们知道,传入函数的第三个参数 param 是一个指向 param_value 地址的指针,所以我们在 printf 中的变量处需要对 param 指针进行解引用,得到一个 uint32_t 类型的变量,对应的占位符使用 %lx 就好啦。
但其实直接对 param 进行解引用是不对的,因为:

  • void * 类型的指针可以指向任何类型的数据,对应地,这样的指针并没有具体的类型信息,所以编译器也不知道它指向的数据的类型和大小。
  • 当需要访问void *指针所指向的数据时,必须先将其转换为具体的指针类型,然后才能解引用。

综上,正确的是第一句。

printf("we got value from event:%lx", *(uint32_t *)param);

原文地址:https://blog.csdn.net/qq_47607743/article/details/139030679

免责声明:本站文章内容转载自网络资源,如本站内容侵犯了原著者的合法权益,可联系本站删除。更多内容请关注自学内容网(zxcms.com)!