【C语言】指针和数组的深入理解(第四期)(下)

简介: 这里在前几期我们已经初略的见识过了,但是这里我们要提一个概念,数组给函数传参是会发生降维的,降维成什么呢?我们看代码:

3、函数指针数组

有了上面的学习就很好理解了,无非就是保存函数地址的数组,那么它的语法格式是什么呢?

int (*arr[10]) (int, int)

这里我们可以分析到:首先 arr 跟 [ ] 先结合,所以它是个数组,这个数组的每个元素是 int (*) (int int) 类型的函数指针,它的作用主要是转移表,那我们这里就简单用一下即可

假设我们需要两个整数的 + - * / 我们写完了四个函数是不是可以放到一个数组里,然后通过访问数组下标就能调用我们想用的函数了:

int add(int x, int y)
{
  return x + y;
}
int sub(int x, int y)
{
  return x - y;
}
int mul(int x, int y)
{
  return x * y;
}
int div(int x, int y)
{
  return x / y;
}
int main()
{
  int (*arr[4]) (int, int) = { add, sub, mul, div };
  printf("加法:%d\n", arr[0](1, 2));
  printf("减法:%d\n", arr[1](5, 2));
  printf("乘法:%d\n", arr[2](3, 3));
  printf("除法:%d\n", arr[3](6, 2));
  return 0;
}

4、指向函数数组的指针

看到这可能有的小伙伴觉得越来越套娃了,但其实这个也很好理解,无非就是一个指针指向了一个数组,数组每个元素是函数指针,这里我们简单了解下概念即可,用的其实也不是很多,当别人如果写了这种代码我们能看懂就行:


函数指针如何定义:

int test(char* str)
{
  if (str == NULL) {
    return 0;
  }
  else
  {
    printf("%s\n", str);
    return 1;
  }
}
int main()
{
  //函数指针pfun
  int (*pfun)(char*) = test;
  //函数指针的数组pfunArr
  int (*pfunArr[5])(char* str);
  pfunArr[0] = test;
  //指向函数指针数组pfunArr的指针ppfunArr
  int (*(*ppfunArr)[5])(char*) = &pfunArr;
  return 0;
}

我们来分析一下这个:int (*(*ppfunArr)[5])(char*),首先看到 (*ppfunArr) 这括号括起来先跟 * 结合证明它是一个指针,指向的类型是什么呢?把它去掉剩下的就是它的类型,int (*[5])(char*),通过这个可以发现,是一个带有5个元素的数组,每个元素的类型是一个函数指针,而函数的返回值为int,参数为 char*

这里我们能看懂即可。

5、回调函数

回调函数指的就是一个通过函数指针调用的函数,如果你把函数的指针(地址),作为参数传递给另一个函数的话,当这个指针被用来调用其指向的函数,这里就被称为回调函数。其实 qsort 函数就是很典型使用了回调函数的例子,感兴趣的可以自行下来了解一下,这里我们就简单的演示下如何使用,用回调函数实现三个数比较大小:

int max(int x, int y, int z, int(*pfun)(int, int))
{
  if (x > pfun(y, z)) {
    return x;
  }
  else
  {
    return pfun(y, z);
  }
}
int tmp(int x, int y) 
{
  return x > y ? x : y;
}
int main()
{
  int ret = max(10, 20, 30, tmp);
  printf("%d\n", ret);
  return 0;
}

比较三个数的最大值是有更优的解决方案的,我们这里只是演示一下回调函数的简单使用,跟上面一样,会用即可,其实不用研究的特别深入

6、一道笔试题

int main()
{
  char* c[] = { "ENTER","NEW","POINT","FIRST" };
  char** cp[] = { c + 3,c + 2,c + 1,c };
  char*** cpp = cp;
  printf("%s\n", **++cpp);
  printf("%s\n", *-- * ++cpp + 3);
  printf("%s\n", *cpp[-2] + 3);
  printf("%s\n", cpp[-1][-1] + 1);
  return 0;
}

这道题我就不讲解了,学习一定得有自己研究的一个过程,包括后续 Java 的文章,每一期基本上都会留一个小疑问让大家自己下去解答,其实这道题很简单,耐心画画图就能理解了,如果你能自己解决这道题,说明你的指针的数组这两章的内容已经通关了,实在是难以解决的话,可以问一下博主。

后续其实还有动态内存管理,但是这个知识点无非就是掌握对 malloc  calloc  realloc  free 的使用,如果你是以后 C++ 方向可学习一下,如果你是 Java 方向其实有个基本认识就行,毕竟 Java接触底层不多,有了前面学习的铺垫,去网上看看内存管理的文章是很轻松学会的,学习最主要是培养学习的能力,

最后来个大总结:从刚开始我们一共讲解了32个关键字,在关键字中也穿插了很多内容,比如大小端,结构体,往后就是符号的理解了,包括我们平常用的注释,以及各种运算符但是除法和取模我们没有放进去,这个在JavaSE系列中会介绍,再往后就是对预处理的深入理解了,最终我们以数组和指针结尾,C语言系列就到此结束了。

相关文章
|
5天前
使用指针访问数组元素
【10月更文挑战第30天】使用指针访问数组元素。
16 3
|
17天前
|
C语言
【c语言】指针就该这么学(1)
本文详细介绍了C语言中的指针概念及其基本操作。首先通过生活中的例子解释了指针的概念,即内存地址。接着,文章逐步讲解了指针变量的定义、取地址操作符`&`、解引用操作符`*`、指针变量的大小以及不同类型的指针变量的意义。此外,还介绍了`const`修饰符在指针中的应用,指针的运算(包括指针加减整数、指针相减和指针的大小比较),以及野指针的概念和如何规避野指针。最后,通过具体的代码示例帮助读者更好地理解和掌握指针的使用方法。
42 0
|
4天前
使用指针访问数组元素
【10月更文挑战第31天】使用指针访问数组元素。
12 2
|
13天前
|
算法 索引
单链表题+数组题(快慢指针和左右指针)
单链表题+数组题(快慢指针和左右指针)
23 1
|
16天前
|
C语言
【c语言】指针就该这么学(3)
本文介绍了C语言中的函数指针、typedef关键字及函数指针数组的概念与应用。首先讲解了函数指针的创建与使用,接着通过typedef简化复杂类型定义,最后探讨了函数指针数组及其在转移表中的应用,通过实例展示了如何利用这些特性实现更简洁高效的代码。
12 2
|
16天前
|
C语言
如何避免 C 语言中的野指针问题?
在C语言中,野指针是指向未知内存地址的指针,可能引发程序崩溃或数据损坏。避免野指针的方法包括:初始化指针为NULL、使用完毕后将指针置为NULL、检查指针是否为空以及合理管理动态分配的内存。
|
16天前
|
C语言
C语言:哪些情况下会出现野指针
C语言中,野指针是指指向未知地址的指针,通常由以下情况产生:1) 指针被声明但未初始化;2) 指针指向的内存已被释放或重新分配;3) 指针指向局部变量,而该变量已超出作用域。使用野指针可能导致程序崩溃或不可预测的行为。
|
23天前
|
存储 C语言
C语言32位或64位平台下指针的大小
在32位平台上,C语言中指针的大小通常为4字节;而在64位平台上,指针的大小通常为8字节。这反映了不同平台对内存地址空间的不同处理方式。
|
22天前
|
存储
如何使用指针数组来实现动态二维数组
指针数组可以用来实现动态二维数组。首先,定义一个指向指针的指针变量,并使用 `malloc` 为它分配内存,然后为每个子数组分配内存。通过这种方式,可以灵活地创建和管理不同大小的二维数组。
|
22天前
|
存储
如何通过指针数组来实现二维数组?
介绍了二维数组和指针数组的概念及其区别,详细讲解了如何使用指针数组模拟二维数组,包括定义与分配内存、访问和赋值元素、以及正确释放内存的步骤,适用于需要动态处理二维数据的场景。