【C语言进阶】qsort函数详解以及它的模拟实现(二)

简介: 【C语言进阶】qsort函数详解以及它的模拟实现(二)

四、利用冒泡排序模拟实现qsort函数

4.1:冒泡排序

 关于冒泡排序的详细讲解可以参考我的这篇文章:初级C语言之【数组】里面详细介绍了冒泡排序

//冒泡排序函数
void bulle_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[10] = { 9,8,7,6,5,4,3,2,1,0 };
  int sz = sizeof(arr) / sizeof(arr[0]);
  bulle_sort(arr,sz);
  int i = 0;
  for (i = 0; i < sz; i++)
  {
    printf("%d ", arr[i]);
  }
  return 0;
}
//结果:
0 1 2 3 4 5 6 7 8 9

缺陷1:

 用来接收待排序数组首元素地址的指针arr已经被写死了,是int*型,说明只能对整型数组进行排序

缺陷2:


9f2f5941b66e45908f91360555fe4c81.png

缺陷2如上图所示,红色方框框起来的部分只适用于对整数之间的大小关系进行比较,然后交换

4.2:模拟实现qsort函数

//利用冒泡排序模拟实现qsort
void Swap(char* ele1, char* ele2,int width)
{
  int i = 0;
  for (i = 0; i < width; i++)
  {
    char tmp = *ele1;
    *ele1 = *ele2;
    *ele2 = tmp;
    ele1++;
    ele2++;
  }
}
void bulle_sort(void* arr , size_t sz,size_t width,int(*comper)(const void*e1,const void*e2))
//第一个参数 - 用来接收待排序数组的首元素地址,可能会排序各种数组,所以形参数组用void*来接收
//第二个参数 - 用来接收数组元素个数
//第三个参数 - 用来接收数组元素的宽度
{
  size_t i = 0;
  //趟数
  for (i = 0; i < sz - 1; i++)
  {
    //一趟冒泡的过程
    size_t j = 0;
    for (j = 0; j < sz-1-i; j++)
    {
      if (comper((char*)arr+j*width,(char*)arr+(j+1)*width)>0)
      //把arr强转为char*,arr就可以正常使用
      //char类型指针+1只会跳过一个字节
      //+j*width表示跳过j个元素
      {
        //交换
        //由于这里的数组名已经被转为char类型的指针
        //所以要交换数组中的元素,就只能一个字节一个字节进行交换
        Swap((char*)arr + j * width, (char*)arr + (j + 1) * width,width);
        //前两个参数是待交换元素的地址,第三个参数是元素的宽度
      }
    }
  }
}

这里说的利用冒泡排序来实现qsort函数,仅仅是实现了qsort函数可以对任意类型的数组进行排序这一特点,并不是说实现qsort函数的底层原理,qsort的底层是通过快速排序来实现的。

 因此,为了使改变之后的冒泡函数能够对任意类型的数组进行排序,原本冒泡排序函数的参数就要发生改变,和qsort函数一样,新的冒泡排序函数也要有以下4个参数:

void* arr

size_t sz

size_t width

int(*comper)(const void*e1,const void*e2)

arr用来接收待排序数组首元素的地址,sz用来接收待排序数组的元素个数,width用来接收数组中每个元素的大小(单位是字节),comper用来接收比较函数的地址。参数的改变解决了原冒泡排序函数的缺陷1。

 接下来要解决原冒泡排序函数的缺陷2,缺陷2的主要问题在于它的普适性不够强,首先要对交换的判断条件,即if后面的判断语句做出改变,让他能够比较任意两个类型的数据,这时,比较函数就发挥作用了,我们只需要把待比较的两个元素地址传给比较函数,由比较函数来判断它们之间的关系

 新的问题又出现了,comper函数的形参需要接收两个待比较元素的地址,这里的待比较元素一定是当前待排序数组里面的元素,但是待排序数组的首元素地址是用空指针(void*)来接收的,无法直接使用,这意味着现在无法通过数组首元素的地址,顺藤摸瓜去访问数组中的元素。这里问题的关键就是空指针无法使用,那我们就想到把空指针进行强制类型转化,把空指针变成有具体指向的指针不就可以正常使用了,问题又出现了,有那么多的指针类型,到底把空指针强转成什么类型的指针呢???答案是:把空指针强转成字符指针(char*)。这里是因为,字符指针+1仅跳过一个字节,我们可以通过改变加数的数值,使指针指向任意内存空间,这也就意味着强转后的arr指针可以存放任意内存单元的地址,并且可通过地址去访问内存中的数据。

(char*)arr+j*width

 这里先把arr指强制类型转化成char*类型,这里的加数是:j*width。其中width表示当前数组中每个元素的大小(单位是字节),这里的+j*width就是跳过j*width个字节,由于width是一个元素的字节,所以+j*width也就意味着跳过了j个元素,此时 (char*)arr+j*width就表示下标为j的元素的地址。

(char*)arr+(j+1)*width

 同理(char*)arr+(j+1)*width就表示下标为j+1的元素的地址。

 此时就可以把待比较的两个元素的地址传给用户自己写的comper函数了,通过comper函数的返回值来判断这两个元素是否要交换。

 到这里缺陷2中的问题只解决了一半,即:只把交换的判断条件做了修改,增强了交换判断条件的普适性,使其可以对任意类型的数组中的元素进行比较。具体的交换步骤还没有修改,当前的交换步骤仅仅适用于整型数据。根据经验,对两个变量进行交换,需要创建一个中间变量,比如:交换两个整型变量,需要创建一个整形的中间变量;交换两个字符型变量,需要创建一个字符型的中间变量……可见,对于不同类型的数据元素,在交换时创建的中间变量的类型也是不同的,由于无法预知要交换数据的类型,所以也无法提前确定中间变量的类型。这里的解决方案是:一个字节一个字节的交换,这样中间变量的类型就能确定下来了,即为char型。不同的数据类型对应着不同的字节数,但是它们都是由字节组成,我们有了数组中每个元素的字节大小时,就可以写一个循环,从元素的首个字节开始,一个字节一个字节的交换,直到最后一个字节。这里我们写了一个交换函数Swap

void Swap(char* ele1, char* ele2,int width)

 Swap函数有三个参数,ele1和ele2分别用来接收待交换的两个元素的地址,width用来接收数组中每个元素的大小(单位是字节)。

void Swap(char* ele1, char* ele2,int width)
{
  int i = 0;
  for (i = 0; i < width; i++)
  {
    char tmp = *ele1;
    *ele1 = *ele2;
    *ele2 = tmp;
    ele1++;
    ele2++;
  }
}

 到此,原来冒泡排序中的两个缺陷已经被成功地解决,经过改造后的冒泡函数bulle_sort就可以对任意类型数组进行排序。

4.3:实际应用

4.3.1:利用bulle_sort函数对整型数组排序:

//利用冒泡排序模拟实现qsort
//交换函数
void Swap(char* ele1, char* ele2,int width)
{
  int i = 0;
  for (i = 0; i < width; i++)
  {
    char tmp = *ele1;
    *ele1 = *ele2;
    *ele2 = tmp;
    ele1++;
    ele2++;
  }
}
//改造后的冒泡排序函数
void bulle_sort(void* arr , size_t sz,size_t width,int(*comper)(const void*e1,const void*e2))
//第一个参数 - 用来接收待排序数组的首元素地址,可能会排序各种数组,所以形参数组用void*来接收
//第二个参数 - 用来接收数组元素个数
//第三个参数 - 用来接收数组元素的宽度
{
  size_t i = 0;
  //趟数
  for (i = 0; i < sz - 1; i++)
  {
    //一趟冒泡的过程
    size_t j = 0;
    for (j = 0; j < sz-1-i; j++)
    {
      if (comper((char*)arr+j*width,(char*)arr+(j+1)*width)>0)
      //把arr强转为char*,arr就可以正常使用
      //char类型指针+1只会跳过一个字节
      //+j*width表示跳过j个元素
      {
        //交换
        Swap((char*)arr + j * width, (char*)arr + (j + 1) * width,width);
        //前两个参数是待交换元素的地址,第三个参数是元素的宽度
      }
    }
  }
}
//比较函数
int comper_int(const void* e1, const void* e2)
{
  return *(int*)e1 - *(int*)e2;
}
//主函数
int main()
{
  int arr[10] = { 9,8,7,6,5,4,3,2,1,0 };
  int sz = sizeof(arr) / sizeof(arr[0]);
  bulle_sort(arr, sz, sizeof(arr[0]), comper_int);//调用bulle_sort来排序
  int i = 0;
  for (i = 0; i < sz; i++)
  {
    printf("%d ", arr[i]);
  }
  printf("\n");
  return 0;
}
//结果:
0 1 2 3 4 5 6 7 8 9

4.3.2:利用bulle_sort函数对结构体数组排序:

按照年龄排序:

//利用冒泡排序模拟实现qsort
//交换函数
void Swap(char* ele1, char* ele2,int width)
{
  int i = 0;
  for (i = 0; i < width; i++)
  {
    char tmp = *ele1;
    *ele1 = *ele2;
    *ele2 = tmp;
    ele1++;
    ele2++;
  }
}
//改造后的冒泡排序函数
void bulle_sort(void* arr , size_t sz,size_t width,int(*comper)(const void*e1,const void*e2))
//第一个参数 - 用来接收待排序数组的首元素地址,可能会排序各种数组,所以形参数组用void*来接收
//第二个参数 - 用来接收数组元素个数
//第三个参数 - 用来接收数组元素的宽度
{
  size_t i = 0;
  //趟数
  for (i = 0; i < sz - 1; i++)
  {
    //一趟冒泡的过程
    size_t j = 0;
    for (j = 0; j < sz-1-i; j++)
    {
      if (comper((char*)arr+j*width,(char*)arr+(j+1)*width)>0)
      //把arr强转为char*,arr就可以正常使用
      //char类型指针+1只会跳过一个字节
      //+j*width表示跳过j个元素
      {
        //交换
        Swap((char*)arr + j * width, (char*)arr + (j + 1) * width,width);
        //前两个参数是待交换元素的地址,第三个参数是元素的宽度
      }
    }
  }
}
//声明一个结构体
struct Stu
{
  char name[20];
  int age;
};
//按照年龄进行比较
int comper_age(const void* e1, const void* e2)
{
  return ((struct Stu*)e1)->age - ((struct Stu*)e2)->age;
}
//主函数
int main()
{
  struct Stu arr[3] = { {"zhangsan",100},{"lisi",20},{"wangwu",3} };
  int sz = sizeof(arr) / sizeof(arr[0]);
  int i = 0;
  //排序前打印
  for (i = 0; i < sz; i++)
  {
    printf("%s %d   ", arr[i].name, arr[i].age);
  }
  printf("\n");
  //按照年龄进行排序
  bulle_sort(arr, sz, sizeof(arr[0]), comper_age);
  //排序后打印
  for (i = 0; i < sz; i++)
  {
    printf("%s %d   ", arr[i].name, arr[i].age);
  }
  printf("\n");
  return 0;
}
//结果:
zhangsan 100   lisi 20   wangwu 3
wangwu 3   lisi 20   zhangsan 100

按照名字排序:

//利用冒泡排序模拟实现qsort
//交换函数
void Swap(char* ele1, char* ele2,int width)
{
  int i = 0;
  for (i = 0; i < width; i++)
  {
    char tmp = *ele1;
    *ele1 = *ele2;
    *ele2 = tmp;
    ele1++;
    ele2++;
  }
}
//改造后的冒泡排序函数
void bulle_sort(void* arr , size_t sz,size_t width,int(*comper)(const void*e1,const void*e2))
//第一个参数 - 用来接收待排序数组的首元素地址,可能会排序各种数组,所以形参数组用void*来接收
//第二个参数 - 用来接收数组元素个数
//第三个参数 - 用来接收数组元素的宽度
{
  size_t i = 0;
  //趟数
  for (i = 0; i < sz - 1; i++)
  {
    //一趟冒泡的过程
    size_t j = 0;
    for (j = 0; j < sz-1-i; j++)
    {
      if (comper((char*)arr+j*width,(char*)arr+(j+1)*width)>0)
      //把arr强转为char*,arr就可以正常使用
      //char类型指针+1只会跳过一个字节
      //+j*width表示跳过j个元素
      {
        //交换
        Swap((char*)arr + j * width, (char*)arr + (j + 1) * width,width);
        //前两个参数是待交换元素的地址,第三个参数是元素的宽度
      }
    }
  }
}
//声明一个结构体
struct Stu
{
  char name[20];
  int age;
};
//按照名字比较
int comper_name(const void* e1, const void* e2)
{
  return strcmp(((struct Stu*)e1)->name, ((struct Stu*)e2)->name);//名字是字符串,所以名字的比较要用到字符串比较 函数strcmp
}
//主函数
int main()
{
  struct Stu arr[3] = { {"zhangsan",100},{"lisi",20},{"wangwu",3} };
  int sz = sizeof(arr) / sizeof(arr[0]);
  int i = 0;
  //排序前打印
  for (i = 0; i < sz; i++)
  {
    printf("%s %d   ", arr[i].name, arr[i].age);
  }
  printf("\n");
  //按照名字进行排序
  bulle_sort(arr, sz, sizeof(arr[0]), comper_name);
  //排序后打印
  for (i = 0; i < sz; i++)
  {
    printf("%s %d   ", arr[i].name, arr[i].age);
  }
  printf("\n");
  return 0;
}
//结果:
zhangsan 100   lisi 20   wangwu 3
lisi 20   wangwu 3   zhangsan 100


目录
相关文章
|
3月前
|
存储 C语言
`scanf`是C语言中用于按格式读取标准输入的函数
`scanf`是C语言中用于按格式读取标准输入的函数,通过格式字符串解析输入并存入指定变量。需注意输入格式严格匹配,并建议检查返回值以确保读取成功,提升程序健壮性。
985 0
|
5月前
|
安全 C语言
C语言中的字符、字符串及内存操作函数详细讲解
通过这些函数的正确使用,可以有效管理字符串和内存操作,它们是C语言编程中不可或缺的工具。
318 15
|
10月前
|
人工智能 Java 程序员
一文彻底搞清楚C语言的函数
本文介绍C语言函数:函数是程序模块化的工具,由函数头和函数体组成,涵盖定义、调用、参数传递及声明等内容。值传递确保实参不受影响,函数声明增强代码可读性。君志所向,一往无前!
384 1
一文彻底搞清楚C语言的函数
|
11月前
|
存储 C语言
【C语言程序设计——函数】递归求斐波那契数列的前n项(头歌实践教学平台习题)【合集】
本关任务是编写递归函数求斐波那契数列的前n项。主要内容包括: 1. **递归的概念**:递归是一种函数直接或间接调用自身的编程技巧,通过“俄罗斯套娃”的方式解决问题。 2. **边界条件的确定**:边界条件是递归停止的条件,确保递归不会无限进行。例如,计算阶乘时,当n为0或1时返回1。 3. **循环控制与跳转语句**:介绍`for`、`while`循环及`break`、`continue`语句的使用方法。 编程要求是在右侧编辑器Begin--End之间补充代码,测试输入分别为3和5,预期输出为斐波那契数列的前几项。通关代码已给出,需确保正确实现递归逻辑并处理好边界条件,以避免栈溢出或结果
567 16
|
11月前
|
存储 编译器 C语言
【C语言程序设计——函数】分数数列求和2(头歌实践教学平台习题)【合集】
函数首部:按照 C 语言语法,函数的定义首部表明这是一个自定义函数,函数名为fun,它接收一个整型参数n,用于指定要求阶乘的那个数,并且函数的返回值类型为float(在实际中如果阶乘结果数值较大,用float可能会有精度损失,也可以考虑使用double等更合适的数据类型,这里以float为例)。例如:// 函数体代码将放在这里函数体内部变量定义:在函数体中,首先需要定义一些变量来辅助完成阶乘的计算。比如需要定义一个变量(通常为float或double类型,这里假设用float。
445 3
|
11月前
|
存储 算法 安全
【C语言程序设计——函数】分数数列求和1(头歌实践教学平台习题)【合集】
if 语句是最基础的形式,当条件为真时执行其内部的语句块;switch 语句则适用于针对一个表达式的多个固定值进行判断,根据表达式的值与各个 case 后的常量值匹配情况,执行相应 case 分支下的语句,直到遇到 break 语句跳出 switch 结构,若没有匹配值则执行 default 分支(可选)。例如,在判断一个数是否大于 10 的场景中,条件表达式为 “num> 10”,这里的 “num” 是程序中的变量,通过比较其值与 10 的大小关系来确定条件的真假。常量的值必须是唯一的,且在同一个。
357 2
|
11月前
|
存储 编译器 C语言
【C语言程序设计——函数】回文数判定(头歌实践教学平台习题)【合集】
算术运算于 C 语言仿若精密 “齿轮组”,驱动着数值处理流程。编写函数求区间[100,500]中所有的回文数,要求每行打印10个数。根据提示在右侧编辑器Begin--End之间的区域内补充必要的代码。如果操作数是浮点数,在 C 语言中是不允许直接进行。的结果是 -1,因为 -7 除以 3 商为 -2,余数为 -1;注意:每一个数据输出格式为 printf("%4d", i);的结果是 1,因为 7 除以 -3 商为 -2,余数为 1。取余运算要求两个操作数必须是整数类型,包括。开始你的任务吧,祝你成功!
197 1
|
存储 C语言
C 语言函数完全指南:创建、调用、参数传递、返回值解析
函数是一段代码块,只有在被调用时才会运行。 您可以将数据(称为参数)传递给函数。 函数用于执行某些操作,它们对于重用代码很重要:定义一次代码,并多次使用。
418 3
|
C语言
C语言函数返回值详解
本文详细解析了C语言中函数返回值的概念与应用。从函数的基本定义入手,深入探讨了不同类型返回值的作用及意义,并提供了实用的编程示例,帮助读者更好地理解和使用函数返回值。通过本文,你将掌握如何有效利用返回值优化代码结构与功能实现。
1200 2
|
C语言
C语言---函数---知识点总结(三)------函数的返回值类型
C语言---函数---知识点总结(三)------函数的返回值类型