【C语言进阶(六)】指针进阶详解(下)(二)

简介: 【C语言进阶(六)】指针进阶详解(下)(二)

3.1 回调函数

先了解一下回调函数的概念
为后面自我实现qsort做准备:

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


4. 库函数中qsort分析

首先qsort是一个用于排序的函数!

先来看官方的定义:

阅读文档可以知道以下内容:

  1. base指针是传待排序的空间
  2. num是指待排序元素个数
  3. size是指每一个元素所占字节大小
  4. 而最后一个参数是一个函数指针

经过仔细查阅资料可得:
函数指针指向的函数需要程序员自己实现
而又两个参数: 我称为A和B

A小于B: 返回值小于一

A大于B: 返回值大于一

A等于B: 返回值等于一


4.1 为什么这个函数需要我们自己实现?

因为每一个类型的数据的比较方式不同
并且操作者想要比较的数据可以有多个

比如:

  1. 整型的比较
int compar(void* x1,void* x2)
{
  return *(int*)x1 - *(*int)x2);
}

对代码的解释:

void*类型的数据不能直接解引用

所以要强制类型转换为int * 类型

假如你想要比较char类型的数据

这里就需要强制转换为char *

  1. 浮点型的比较
int compar(void* x1,void* x2)
{
  return strcmp( (char*)x1, (char*)x2 );
}

对代码的解释:

字符不能单纯使用大于小于号比较

应该使用字符串操作函数比较大小

  1. 结构体的比较
struct stu 
{
  char name[20]={0};//学生姓名
  int height =0;//学生身高
  int grade =0;//学生成绩
};
compar(void* x1,void* x2)//以学生身高作为比较基准
{
  return ((struct stu*)x1)->height -((struct stu*)x2)->height;
         //强制转换为结构体指针   比较身高,指向height
}
compar(void* x1,void* x2)//以学生名字作为比较基准
{
  return strcmp(((struct stu*)x1)->name,((struct stu*)x2)->name);
               强制类型转换为结构体指针  指向结构体中的name数组
}

对代码的解释:

在结构体中可以比较不同的成员

而比较前都应该先进行强制类型转换

并且指向对应的结构体成员


4.2 库函数qsort的使用

假设我们需要排序一个整型数组

下面来使用一下qsort函数

int compar(void* x1,void*x2)
{
  return *(int*)x1 - *(int*)x2;
}
int main()
{
  int a[6]={6,2,9,4,3,5};
  qsort(a, 6, sizeof(int), compar);
  return 0;
}

5. qsort函数的模拟实现

由于库函数中的qsort函数
的内部是由快速排序实现的
比较难理解
所以我们把内部简化为冒泡排序!

基本框架:

  1. 两层 if 语句实现冒泡大框架
  2. 比较大小时使用自定义函数compar
  3. 交换函数使用自定义函数Swap

5.1 大框架的实现

bubble_qsort(void* s, int sz, int width, int (*cmp)(void* x1, void* x2))
{
  for (int i = 0; i < sz - 1; i++)
  {
    for (int j = 0; j < sz - i - 1; j++)
    {
      if (比较条件函数)
      {
        Swap函数
      }
    }
  }
}

模仿库函数中的参数
用冒泡函数原理自己实现一个qsort!


5.2 比较函数的实现

在本章的4.2部分已经模拟了很多情况

假设我们想要排序整型数组:

int cmp(void* x1, void* x2)//整数比较
{
  return *(int*)x1 - *(int*)x2;
}

5.3 对于交换函数的思考

在使用冒泡排序时
我们通常已知待排序的数组的类型
所以能够使用不同的方法来交换不同类型

然而

qsort函数需要满足所有类型的交换

解决方法:

可以把数据一个字节一个字节的交换
任何类型的大小都大于等于一个字节
所以只需要知道数据所占字节数
就能解决这个问题!


5.4 交换函数的实现

void Swap_by_bite(char* s1, char* s2, int width)
{
  for (int i = 0; i < width; i++)//一个字节一个字节的拷贝过去
  {
    char tmp = *s1;
    *s1 = *s2;
    *s2 = tmp;
    s1++;
    s2++;
  }
}

对代码的解释:

函数参数使用char*是因为
将数据单位化,方便以字节为单位交换
width参数是指数据所占字节大小


5.5 qsort函数所有代码

int cmp(void* x1, void* x2)//整数比较
{
  return *(int*)x1 - *(int*)x2;
}
void Swap_by_bite(char* s1, char* s2, int width)
{
  for (int i = 0; i < width; i++)//一个字节一个字节的拷贝过去
  {
    char tmp = *s1;
    *s1 = *s2;
    *s2 = tmp;
    s1++;
    s2++;
  }
}
bubble_sort(void* s, int sz, int width, int (*cmp)(void* x1, void* x2))
{
  for (int i = 0; i < sz - 1; i++)
  {
    for (int j = 0; j < sz - i - 1; j++)
    {
      if (cmp((char*)s+j*width,(char*)s+(j+1)*width)>0)
      {
        Swap_by_bite((char*)s + j * width, (char*)s + (j + 1) * width, width);
      }
    }
  }
}

对代码的解释:

给cmp函数传参:s+j*width和

s+(j+1)*width可以这样理解:

而交换函数传参也是一个意思

并且Swap函数的参数是char*

所以需要我们强制类型转换传入的数据


6. 总结

指针是C语言的一项利器
某些使用C语言编写的项目
使用高阶指针解决问题是家常便饭!
看似这种不断套娃的过程很鸡肋
实际上有一定的用途


🔎 下期预告:自定义类型详解 🔍



相关文章
|
2月前
|
存储 C语言
【C语言篇】深入理解指针3(附转移表源码)
【C语言篇】深入理解指针3(附转移表源码)
42 1
|
2月前
|
C语言
【c语言】指针就该这么学(1)
本文详细介绍了C语言中的指针概念及其基本操作。首先通过生活中的例子解释了指针的概念,即内存地址。接着,文章逐步讲解了指针变量的定义、取地址操作符`&`、解引用操作符`*`、指针变量的大小以及不同类型的指针变量的意义。此外,还介绍了`const`修饰符在指针中的应用,指针的运算(包括指针加减整数、指针相减和指针的大小比较),以及野指针的概念和如何规避野指针。最后,通过具体的代码示例帮助读者更好地理解和掌握指针的使用方法。
52 0
|
12天前
|
存储 程序员 编译器
C 语言数组与指针的深度剖析与应用
在C语言中,数组与指针是核心概念,二者既独立又紧密相连。数组是在连续内存中存储相同类型数据的结构,而指针则存储内存地址,二者结合可在数据处理、函数传参等方面发挥巨大作用。掌握它们的特性和关系,对于优化程序性能、灵活处理数据结构至关重要。
|
2月前
|
C语言
【c语言】指针就该这么学(3)
本文介绍了C语言中的函数指针、typedef关键字及函数指针数组的概念与应用。首先讲解了函数指针的创建与使用,接着通过typedef简化复杂类型定义,最后探讨了函数指针数组及其在转移表中的应用,通过实例展示了如何利用这些特性实现更简洁高效的代码。
20 2
|
2月前
|
C语言
如何避免 C 语言中的野指针问题?
在C语言中,野指针是指向未知内存地址的指针,可能引发程序崩溃或数据损坏。避免野指针的方法包括:初始化指针为NULL、使用完毕后将指针置为NULL、检查指针是否为空以及合理管理动态分配的内存。
|
2月前
|
C语言
C语言:哪些情况下会出现野指针
C语言中,野指针是指指向未知地址的指针,通常由以下情况产生:1) 指针被声明但未初始化;2) 指针指向的内存已被释放或重新分配;3) 指针指向局部变量,而该变量已超出作用域。使用野指针可能导致程序崩溃或不可预测的行为。
|
2月前
|
存储 C语言
C语言32位或64位平台下指针的大小
在32位平台上,C语言中指针的大小通常为4字节;而在64位平台上,指针的大小通常为8字节。这反映了不同平台对内存地址空间的不同处理方式。
|
2月前
|
存储 算法 C语言
C语言:什么是指针数组,它有什么用
指针数组是C语言中一种特殊的数据结构,每个元素都是一个指针。它用于存储多个内存地址,方便对多个变量或数组进行操作,常用于字符串处理、动态内存分配等场景。
|
2月前
|
存储 C语言
C语言指针与指针变量的区别指针
指针是C语言中的重要概念,用于存储内存地址。指针变量是一种特殊的变量,用于存放其他变量的内存地址,通过指针可以间接访问和修改该变量的值。指针与指针变量的主要区别在于:指针是一个泛指的概念,而指针变量是具体的实现形式。
|
2月前
|
C语言
C语言指针(3)
C语言指针(3)
14 1