C语言 指针进阶 (下)

简介: C语言 指针进阶

5.函数指针


指向函数地址的指针
int(*pf)(int,int)
pf为指针,指向参数为int 的函数,返回类型是int


#include<stdio.h>
void test()
{}
int main()
{
  test();
  printf("%p\n", &test);
  printf("%p\n", test);
  return 0;
}


f8672dde59af5caba93d3473b16a1dc6_948152648d31468693f5ce694f93a635.png


对于函数:&函数名和函数名都是函数的地址


#include<stdio.h>
int Add(int x, int y)
{
  return x + y;
}
int main()
{
  int(*p)(int, int) = &Add;
  //p存放函数Add的地址,解引用得到函数名
  //进行传参调用
  int ret = (*p)(2, 3);
  printf("%d\n", ret);
  return 0;
}


54dcd06ce7b3b59abb3f8ebc7890d4dd_f0dc95144419473e9eaf169c7ff5e5be.png


#include<stdio.h>
int Add(int x, int y)
{
  int z = x + y;
  return z;
}
int main()
{
  int(*p)(int, int) = Add;
  printf("%d\n", p(2, 3));
  printf("%d\n", Add(2, 3));
  printf("%d\n", (*p)(2, 3));
  return 0;
}


函数指针的函数调用

*p(2,3)==(p)(2,3)


函数指针数组

存放函数地址的数组


#include<stdio.h>
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(*parr[4])(int, int) = { Add,Sub,Mul,Div };
  int i = 0;
  for (i = 0; i < 4; i++)
  {
  printf("%d\n", (*parr)[i](2, 3));
  }
  return 0;
}


习题


#include<stdio.h>
char* my_strcpy(char*dest,const char*src)
//写函数指针,能够指向my_strcpy
//写函数指针数组,能够存放4个my_strcpy函数的地址
char* (*pf)(char*dest,const char*src)
char* (*pfarr[4])(char*dest,const char*src)


计算器

1.switch函数


```c
#include<stdio.h>
void menu()
{
  printf("*********************************************\n");
  printf("******  1.add      2.sub               ******\n");
  printf("******  3.mul      4.div               ******\n");
  printf("************   0.exit             ***********\n");
  printf("*********************************************\n");
}
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;
}
void Calc(int(*pf)(int, int))
{
  int x = 0;
  int y = 0;
  printf("请输入两个操作数");
  scanf("%d%d", &x, &y);
  printf("%d\n", pf(x, y));
}
int main()
{
  int input = 0;
  do
  {
  menu();
  printf("请选择");
  scanf("%d", &input);
  switch (input)
  {
  case 1:
    Calc(Add);
    break;
  case 2:
    Calc(Sub);
    break;
  case 3:
    Calc(Mul);
    break;
  case 4:
    Calc(Div);
    break;
  case 0:
    printf("退出");
  default:
    printf("选择错误");
    break;
  }
  } while (input);
  return 0;
}


2,函数指针数组


#include<stdio.h>
void menu()
{
  printf("*********************************************\n");
  printf("******  1.add      2.sub               ******\n");
  printf("******  3.mul      4.div               ******\n");
  printf("************   0.exit             ***********\n");
  printf("*********************************************\n");
}
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 input = 0;
  int x = 0;
  int y = 0;
  int(*pfarr[5])(int, int) = { 0,Add,Sub,Mul,Div };
  do
  {
  menu();
  printf("请选择");
  scanf("%d", &input);
  if (input >= 1 && input <= 4)
  {
    printf("请输入两个操作数");
    scanf("%d%d", &x, &y);
    int ret = pfarr[input](x, y);
    printf("%d\n", ret);
  }
  else if (input == 0)
  {
    printf("退出");
  }
  else
  {
    printf("选择错误");
  }
  } while (input);
  return 0;
}


回调函数

回调函数是通过函数指针调用的函数

如果把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,这就是回调函数。

void类型的指针,可以接受任意类型的地址

void类型的指针,不能进行解引用操作

void*类型的指针,不能进行加减整数的运算


冒泡排序


#include<stdio.h>
void bubble_sort(int arr[], int sz)
{
  int i = 0;
  //确定趟数
  for (i = 0; i < sz - 1; i++)
  {
  int j = 0;
  //每一趟所比较的次数
  for (j = 0; j < sz - 1 - i; j++)
  {
    if (arr[j] > arr[j + 1])
    {
    int tmp = arr[j];
    arr[j] = arr[j + 1];
    arr[j + 1] = tmp;
    }
  }
  }
}
int main()
{
  int arr[] = { 9,8,7,6,5,4,3,2,1,0 };
  int sz = sizeof(arr) / sizeof(arr[0]);
  bubble_sort(arr, sz);
  int i = 0;
  for (i = 0; i < sz; i++)
  {
  printf("%d ", arr[i]);
  }
  return 0;
}


qsort函数可实现任意类型数值的比较(数组)


void qsort(void*base,size num,size width,int(*cmp)(const void*e1,const void*e2))
// 第一个参数,待排序数组的首元素地址
// 第二个参数,待排序数组的元素个数
// 第三个参数,待排序数组的每个元素的大小-单位为字节
// 第四个参数,函数指针,比较两个元素的所用函数的地址
// 函数指针的两个参数:待比较的两个元素地址


qsort函数用于冒泡排序(数组)


#include<stdio.h>
#include<stdlib.h>
int cmp_int(void* e1,void* e2)
{
  //比较两个整型值
  return *(int*)e1 - *(int*)e2;
}
void Swap(char* c1, char* c2, int width)
{
  int i = 0;
  for (i = 0; i < width; i++)
  {
  char tmp = *c1;
  *c1 = *c2;
  *c2 = tmp;
  c1++;
  c2++;
  }
}
void bubble_sort(void* base, int sz, int width, int(*cmp)(void* e1, void* e2))
{
  int i = 0;
  //趟数
  for (i = 0; i < sz - 1; i++)
  {
  //每一趟比较的对数
  int j = 0;
  for (j = 0; j < sz - 1 - i; j++)
  {
    //两个元素的比较
    if (cmp((char*)base + j * width, (char*)base + (j + 1) * width) > 0)
    {
    //交换
    Swap((char*)base + j * width, (char*)base + (j + 1) * width, width);
    }
  }
  }
}
void test()
{
  int arr[] = { 9,8,7,6,5,4,3,2,1,0 };
  int sz = sizeof(arr) / sizeof(arr[0]);
  bubble_sort(arr, sz, sizeof(arr[0]), cmp_int);
  int i = 0;
  for (i = 0; i < sz; i++)
  {
  printf("%d ", arr[i]);
  }
}
int main()
{
  test();
  return 0;
}


指向函数指针数组的指针


int (*pfArr[4])(int, int);//pfArr是数组,,函数指针的数组
int(*(*ppfArr)[4])(int, int) = &pfArr;
//ppfArr是数组指针,指针指向的数组有4个元素
//指向的数组的每个元素的类型是一个函数指针 int(*)(int,int)

目录
相关文章
|
C语言
【c语言】指针就该这么学(1)
本文详细介绍了C语言中的指针概念及其基本操作。首先通过生活中的例子解释了指针的概念,即内存地址。接着,文章逐步讲解了指针变量的定义、取地址操作符`&`、解引用操作符`*`、指针变量的大小以及不同类型的指针变量的意义。此外,还介绍了`const`修饰符在指针中的应用,指针的运算(包括指针加减整数、指针相减和指针的大小比较),以及野指针的概念和如何规避野指针。最后,通过具体的代码示例帮助读者更好地理解和掌握指针的使用方法。
230 1
|
11月前
|
存储 NoSQL 编译器
【C语言】指针的神秘探险:从入门到精通的奇幻之旅 !
指针是一个变量,它存储另一个变量的内存地址。换句话说,指针“指向”存储在内存中的某个数据。
368 7
【C语言】指针的神秘探险:从入门到精通的奇幻之旅 !
|
9月前
|
存储 人工智能 Java
一文轻松拿捏C语言的指针的基础使用
本文介绍了C语言中的指针概念,包括直接访问和间接访问内存的方式、指针变量的定义与使用、取址运算符`&`和取值运算符`*`的应用,帮助读者深入理解指针这一C语言的核心概念。君志所向,一往无前!
176 0
|
11月前
|
存储 编译器 C语言
【C语言】指针大小知多少 ?一场探寻C语言深处的冒险 !
在C语言中,指针的大小(即指针变量占用的内存大小)是由计算机的体系结构(例如32位还是64位)和编译器决定的。
1307 9
|
11月前
|
安全 程序员 C语言
【C语言】指针的爱恨纠葛:常量指针vs指向常量的指针
在C语言中,“常量指针”和“指向常量的指针”是两个重要的指针概念。它们在控制指针的行为和数据的可修改性方面发挥着关键作用。理解这两个概念有助于编写更安全、有效的代码。本文将深入探讨这两个概念,包括定义、语法、实际应用、复杂示例、最佳实践以及常见问题。
353 7
|
存储 C语言
C语言如何使用结构体和指针来操作动态分配的内存
在C语言中,通过定义结构体并使用指向该结构体的指针,可以对动态分配的内存进行操作。首先利用 `malloc` 或 `calloc` 分配内存,然后通过指针访问和修改结构体成员,最后用 `free` 释放内存,实现资源的有效管理。
1097 13
|
存储 C语言 开发者
C 语言指针与内存管理
C语言中的指针与内存管理是编程的核心概念。指针用于存储变量的内存地址,实现数据的间接访问和操作;内存管理涉及动态分配(如malloc、free函数)和释放内存,确保程序高效运行并避免内存泄漏。掌握这两者对于编写高质量的C语言程序至关重要。
383 11
|
12月前
|
存储 算法 程序员
C 语言指针详解 —— 内存操控的魔法棒
《C 语言指针详解》深入浅出地讲解了指针的概念、使用方法及其在内存操作中的重要作用,被誉为程序员手中的“内存操控魔法棒”。本书适合C语言初学者及希望深化理解指针机制的开发者阅读。
|
12月前
|
存储 程序员 编译器
C 语言数组与指针的深度剖析与应用
在C语言中,数组与指针是核心概念,二者既独立又紧密相连。数组是在连续内存中存储相同类型数据的结构,而指针则存储内存地址,二者结合可在数据处理、函数传参等方面发挥巨大作用。掌握它们的特性和关系,对于优化程序性能、灵活处理数据结构至关重要。
|
12月前
|
算法 C语言
C语言中的文件操作技巧,涵盖文件的打开与关闭、读取与写入、文件指针移动及注意事项
本文深入讲解了C语言中的文件操作技巧,涵盖文件的打开与关闭、读取与写入、文件指针移动及注意事项,通过实例演示了文件操作的基本流程,帮助读者掌握这一重要技能,提升程序开发能力。
709 3