[C语言] 指针的进阶(下)

文章目录

前言

一、函数指针

二、函数指针数组

三、回调函数 

总结


前言

C语言中指针的重要性是:通过指针不仅可以对数据本身,还可以对存储数据的变量地址进行操作。指针能够帮助我们快速地传递数据,减少内存的耗用,还能够方便我们处理字符串。


一、函数指针

首先看一段代码:

#include <stdio.h>
void test()
{
 printf("hehe\n");
}
int main()
{
 printf("%p\n", test);
 printf("%p\n", &test);
 return 0;
}
//下面pfun1和pfun2哪个有能力存放test函数的地址?
void (*pfun1)();
void *pfun2();

 

可见,输出的两个地址 都是test函数的地址。 那函数的地址想要保存起来,就要求pfun1或者pfun2是指针,哪个是指针:

pfun1可以存放。pfun1先和*结合,说明pfun1是指针,指针指向的是一个函数,指向的函数无参
数,返回值类型为void。

 

函数指针是一种指针类型,它可以指向具有相同返回值类型和参数列表的函数。函数指针的用法有以下几个方面:

  • - 函数指针的定义:函数指针的定义方式为:`返回值类型 (*指针变量名) (参数列表);`,例如:`int (*fp) (int a, int b);`表示定义了一个指向返回值为int,参数为两个int的函数的指针fp。
  • - 函数指针的赋值:函数指针可以用函数名(不带括号和参数)来赋值,表示指向该函数的地址,例如:`fp = max;`表示让fp指向max函数。
  • - 函数指针的调用:函数指针可以用`(*指针变量名) (实参列表)`的形式来调用所指向的函数,例如:`(*fp) (10, 20);`表示调用fp所指向的函数,并传入10和20作为实参。
  • - 函数指针作为参数:函数指针可以作为另一个函数的参数,从而实现回调函数的功能,即把一个函数作为另一个函数的输入,例如:`void sort(int arr[], int n, int (*cmp) (int a, int b));`表示定义了一个排序函数,它接受一个数组,一个数组长度,和一个比较函数作为参数。
  • - 函数指针作为返回值:函数指针也可以作为另一个函数的返回值,从而实现根据不同条件返回不同函数的功能,例如:`int (*get_op(char op)) (int a, int b);`表示定义了一个获取运算符对应函数的函数,它接受一个字符作为参数,返回一个对应的运算函数。

函数指针是一种指向函数的指针变量,它可以用来间接调用函数。函数指针的调用方法有两种:

- 一种是直接使用函数指针变量名加上参数列表,例如 `fp(9);`,这里的 `fp` 是一个函数指针变量,它指向一个接受一个 `int` 参数并返回 `int` 值的函数。
- 另一种是使用 `*` 运算符解引用函数指针变量,然后加上参数列表,例如 `(*fp)(9);`,这里的 `*fp` 表示函数指针变量

二、函数指针数组

数组是一个存放相同类型数据的存储空间:

int *arr[10];
//数组的每个元素是int*

 那要把函数的地址存到一个数组中,那这个数组就叫函数指针数组,那函数指针的数组如何定义呢? 

int (*parr1[10])();
int *parr2[10]();
int (*)() parr3[10];

 答案是:parr1

parr1 先和 [] 结合,说明 parr1是数组,数组的内容是什么呢?
是 int (*)() 类型的函数指针。
函数指针数组的用途: 转移表  例如计算器
#include <stdio.h>
int add(int a, int b)
{
           return a + b;
}
int sub(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 main()
{     int x, y;
     int input = 1;
     int ret = 0;
     int(*p[5])(int x, int y) = { 0, add, sub, mul, div }; //转移表
     while (input)
     {
          printf( "*************************\n" );
          printf( " 1:add           2:sub \n" );
          printf( " 3:mul           4:div \n" );
          printf( "*************************\n" );
          printf( "请选择:" );
      scanf( "%d", &input);
          if ((input <= 4 && input >= 1))
         {
          printf( "输入操作数:" );
              scanf( "%d %d", &x, &y);
              ret = (*p[input])(x, y);
         }
          else
               printf( "输入有误\n" );
          printf( "ret = %d\n", ret);
     }
      return 0;
}

三、回调函数 

回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个 函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应。

 首先演示一下qsort函数的使用:

#include <stdio.h>
//qosrt函数的使用者得实现一个比较函数
int int_cmp(const void * p1, const void * p2)
{
  return (*( int *)p1 - *(int *) p2);
}
int main()
{
    int arr[] = { 1, 3, 5, 7, 9, 2, 4, 6, 8, 0 };
    int i = 0;
    
    qsort(arr, sizeof(arr) / sizeof(arr[0]), sizeof (int), int_cmp);
    for (i = 0; i< sizeof(arr) / sizeof(arr[0]); i++)
   {
       printf( "%d ", arr[i]);
   }
    printf("\n");
    return 0;
}

 这里使用回调函数,模拟实现qsort(采用冒泡的方式)。

注意:这里第一次使用 void* 的指针,讲解 void* 的作用。

#include <stdio.h>
int int_cmp(const void * p1, const void * p2)
{
  return (*( int *)p1 - *(int *) p2);
}
void _swap(void *p1, void * p2, int size)
{
    int i = 0;
    for (i = 0; i< size; i++)
   {
        char tmp = *((char *)p1 + i);
       *(( char *)p1 + i) = *((char *) p2 + i);
       *(( char *)p2 + i) = tmp;
   }
}
void bubble(void *base, int count , int size, int(*cmp )(void *, void *))
{
    int i = 0;
    int j = 0;
  for (i = 0; i< count - 1; i++)
   {
       for (j = 0; j<count-i-1; j++)
       {
            if (cmp ((char *) base + j*size , (char *)base + (j + 1)*size) > 0)
           {
               _swap(( char *)base + j*size, (char *)base + (j + 1)*size, size);
           }
       }
   }
}
int main()
{
    int arr[] = { 1, 3, 5, 7, 9, 2, 4, 6, 8, 0 };
    //char *arr[] = {"aaaa","dddd","cccc","bbbb"};
    int i = 0;
    bubble(arr, sizeof(arr) / sizeof(arr[0]), sizeof (int), int_cmp);
    for (i = 0; i< sizeof(arr) / sizeof(arr[0]); i++)
   {
       printf( "%d ", arr[i]);
   }
    printf("\n");
    return 0;
}

 回调函数的使用是对函数指针的应用,函数指针的概念本身很简单,但是把函数指针应用于回调函数就体现了一种解决问题的策略,一种设计系统的思想。


总结

以上就是指针的进阶部分。

  • 5
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

吉始

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值