C语言中qsort函数用法及用冒泡排序实现

 更新时间:2022年10月27日 10:52:52   作者:蜗牛牛啊  
qsort函数是由C语言提供的标准库函数, 它的实现思想是快速排序。这篇文章主要介绍了C语言中qsort函数用法及用冒泡排序实现qsort函数功能,需要的可以参考一下

一、qsort函数简介

qsort函数是由C语言提供的标准库函数, 它的实现思想是快速排序,qosrt函数的头文件是stdlib.h

qsort函数可以排序任意类型的数组。它的返回类型是void,参数是(void* base,size_t num,size_t size,int(*compar)(const void* e1,const void* e2))

接下来对它的参数进行分析:

参数void* base中base是void*类型的指针,base指向待排序数组的第一个元素的地址,也就是指向数组的起始位置。

参数size_t num中size_t表示的是无符号整型,num表示的是待排序数组中总的元素个数。

参数size_t size中size表示的是待排序数组中每个元素占几个字节。

参数int (*compar)(const void* e1, const void* e2)是函数指针,int表示指向函数的返回类型是int类型,compar表示的是函数指针变量,指向函数的地址,被指向的函数用于比较两个元素的大小。

在参数int (*compar)(const void* e1, const void* e2)中,函数指针变量compar指向的函数返回类型是int,返回值的大小可以分为三种情况:

返回值条件
>0e1>e2
<0e1<e2
=0e1=e2

二、qsort函数的使用

1.整型数组排序

#include <stdio.h>
#include <stdlib.h>
int cmp_int(const void* e1, const void* e2)
{
    return *((int*)e1) - *((int*)e2);//默认是升序,将return中e1和e2的位置交换可以使其降序
    //无法对void*类型的指针进行具体操作,因此要将其强制类型转换为int*,使其能够在解引用时访问到4个字节
}
int main()
{
    int i = 0;
    int arr[] = { 7,8,9,4,5,6,1,2,3,10 };
    int sz = sizeof(arr) / sizeof(arr[0]);//计算整型数组中有多少个元素
    qsort(arr, sz, sizeof(int), cmp_int);//sizeof(int)是用来计算int类型的数据所占字节数
    for (i = 0; i < sz; i++)
    {
        printf("%d ", arr[i]);//输出为1 2 3 4 5 6 7 8 9 10
    }
    return 0;
}

注意:当函数的形参为void*类型,要对其进行具体操作时,应该将其强制类型转换为所需的指针类型

2.字符串排序

#include <stdio.h>
#include <stdlib.h>
int cmp_char(const void* e1, const void* e2)//字符串排序
{
    return strcmp((char*)e1, (char*)e2);//比较字符大小用strcmp
}
int main()
{
    char arr[] = "badecf";
    int i = 0;
    //int sz = sizeof(arr) / sizeof(arr[0]);//sizeof求字符串长度会将\0算进去,也就是实际计算出来的会比数组中的多1。
    int sz = strlen(arr);//strlen是用来计算字符串长度的,且只能用来计算字符串的长度。
    qsort(arr,sz , sizeof(char), cmp_char);
    printf("%s\n", arr);//abcdef
    return 0;
}

3.字符串长度排序

#include <stdio.h>
#include <stdlib.h>
int cmp_char_strlen(const void* e1, const void* e2)//比较字符串长度
{
    return strlen(*(char**)e1) - strlen(*(char**)e2);//先将e1和e2强制类型转换成为char**类型,再解引用取出数组中字符串的地址,计算字符串长度
}
int main()
{
    //char* arr[] = {"abcdef" "abc","defs" };
    //也可以通过下面的方式来写指针数组,下面的4行和上面的这一行作用一样
    char* arr1 = "abc";
    char* arr2 = "abcd";
    char* arr3 = "abcdef";
    char* arr[] = { arr1,arr2,arr3 };
    int i = 0;
    qsort(arr, 3, sizeof(char*), cmp_char_strlen);
    for (i = 0; i < 3; i++)
    {
        printf("%s\n", arr[i]);//输出结果为 abc  defs  abcdef
    }
    return 0;
}

4.浮点型数组排序

#include <stdio.h>
#include <stdlib.h>
int cmp_float(const void* e1, const void* e2)//浮点型数组排序
{
    //return (int)(*(float*)e1 - *(float*)e2);//可能会出错,如12.002和12.001比较之后再强制类型转换为int类型后,结果为0
    if (*(float*)e1 - *(float*)e2 > 0.000000)
    {
        return 1;//当e1>e2时,返回大于0的数
    }
    if (*(float*)e1 - *(float*)e2 < 0.000000)
    {
        return -1;//当e1<e2时,返回小于0的数
    }
    else
        return 0;//当e1和e2相等时,返回等于0的数
}
int main()
{
    float arr[3] = { 12.01f,12.03f,12.09f };
    int sz = sizeof(arr) / sizeof(arr[0]);//计算数组长度
    qsort(arr, sz, sizeof(float), cmp_float);
    int i = 0;
    for (i = 0; i < sz; i++)
    {
        printf("%f  ", arr[i]);//输出结果为12.010000  12.030000  12.090000
    }
    return 0;
}

5.结构体类型排序

#include <stdio.h>
#include <stdlib.h>
struct stu
{
    char name[20];
    int age;
};
int cmp_s_age(const void* e1, const void* e2)//比较结构体中int类型
{
    return ((struct stu*)e1)->age - ((struct stu*)e2)->age;
}
int cmp_s_name(const void* e1, const void* e2)//比较结构体中char类型的数组
{
    return strcmp(((struct stu*)e1)->name, ((struct stu*)e2)->name);
}
int main()
{
    struct stu s[] = { {"zhangsan",21} ,{"lisi",20},{"wangwu",52}};
    int i = 0;
    int sz = sizeof(s) / sizeof(s[0]);
    for (i = 0; i < sz; i++)
    {
        printf("%s %d\n", s[i].name, s[i].age);
    }
    qsort(s, sz, sizeof(struct stu), cmp_s_age);//比较年龄,即比较int类型
    //qsort(s, sz, sizeof(struct stu), cmp_s_name);//比较姓名,即比较char类型的数组
    printf("\n");
    for (i = 0; i < sz; i++)
    {
        printf("%s %d\n", s[i].name, s[i].age);//输出为年龄按照升序排列,如果比较姓名,则按照姓名的首字符的ASCII值按照升序排列
    }
    return 0;
}

三、冒泡排序实现qsort函数的功能

1.冒泡排序简介

冒泡排序是通过对两个相邻元素进行比较,如果不符合条件就相互交换位置,并与后面相邻的元素再次比较,直至满足条件为止。

对于冒泡排序,如果它有n个元素,它将进行n-1次的循环。冒泡排序的缺点是只能对整型数组进行排序。

我们通过对整型数组排序进一步了解冒泡排序

//用冒泡排序对整型数组进行排序
#include <stdio.h>
int main()
{
    int i = 0;
    int j = 0;
    int arr[] = { 7,8,9,4,5,6,1,2,3,10 };
    int sz = sizeof(arr) / sizeof(arr[0]);
    for (i = 0; i < sz-1; i++)//总共循环sz-1次
    {
        int flag = 1;//标记,假设数组是有序的
        for (j = 0; j < sz - 1 - i; j++)//第一个元素需要和剩余的sz-1个元素比较,第二个元素需要和sz-1-1个元素比较……所以每次循环需要比较的次数为sz-1-i
        {
            if (arr[j] < arr[j + 1])//从大到小排列
            {
                int tmp = arr[j + 1];
                arr[j + 1] = arr[j];
                arr[j] = tmp;
                flag = 0;//倘若发生交换说明数组是无序的
            }
        }
        if(flag == 1)
        {
            break;//如果没发生一次交换说明数组已经是符合要求的有序数组
        }
    }
    for (i = 0; i < sz; i++)
    {
        printf("%d ", arr[i]);//输出结果为 10 9 8 7 6 5 4 3 2 1
    }
    return 0;
}

2.冒泡排序实现qsort函数功能

对冒泡排序进行qsort化时,要参照qsort函数的参数,尽量做到参数一致。

void qsort_bubble(void* base, int sz, int width, int(*cmp)(const void* e1, const void* e2))

void* base表示对应数组元素的起始位置。

int sz表示数组中元素总个数。

int width表示数组中每个元素所占字节数。

int(*cmp)(const void* e1, const void* e2)是函数指针,指向比较两个元素大小的函数。

设计的冒泡排序qsort化函数qsort_bubble与库函数qsort的参数意义相同。在qsort_bubble和qsort参数中函数指针指向的用于比较数组中两个元素大小的函数不用发生改变,可以直接引用。

利用冒泡排序思想实现qsort函数功能并对结构体排序。

#include <stdio.h>
struct stu
{
	char name[20];
	int age;
};
int cmp_s_age(const void* e1, const void* e2)//比较结构体中整型
{
	return ((struct stu*)e1)->age - ((struct stu*)e2)->age;
}
int cmp_s_name(const void* e1, const void* e2)//比较结构体中整型
{
	return strcmp(((struct stu*)e1)->name, ((struct stu*)e2)->name);
}
void Swap(char* buf1, char* buf2, int width)
{
	int i = 0;
	for (i = 0; i < width; i++)//一个字节一个字节的交换
	{
		char tmp = *buf1;
		*buf1 = *buf2;
		*buf2 = tmp;
		buf1++;
		buf2++;
	}
}
void qsort_bubble(void* base, int sz, int width, int(*cmp)(const void* e1, const 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)
				//强制转换成char*类型的指针,进行交换时一个字节一个字节的交换,+width,向后走对应的字节
			{
				Swap((char*)base + j * width, (char*)base + (j + 1) * width,width);//满足条件进行交换
			}
		}
	}
}
int main()
{
	struct stu s[] = { {"zhangsan",21} ,{"lisi",20},{"wangwu",52}};
	int i = 0;
	int sz = sizeof(s) / sizeof(s[0]);
	//printf("%d", sz);
	for (i = 0; i < sz; i++)
	{
		printf("%s %d\n", s[i].name, s[i].age);
	}
	qsort_bubble(s, sz, sizeof(struct stu), cmp_s_age);//比较年龄,即比较int类型
	//qsort_bubble(s, sz, sizeof(struct stu), cmp_s_name);//比较姓名,即比较char类型的数组
	printf("\n");
	for (i = 0; i < sz; i++)
	{
		printf("%s %d\n", s[i].name, s[i].age);
	}
	return 0;
}

到此这篇关于C语言中qsort函数用法及用冒泡排序实现的文章就介绍到这了,更多相关C语言 qsort函数内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • C++中的并行与并发基础与使用详解

    C++中的并行与并发基础与使用详解

    对于多线程来说,这两个概念有很大部分是重叠的。对于很多人来说,它们的意思没有什么区别。其区别主要在于关注点和意图方面(差距甚微)。这两个词都是用来对硬件在同时执行多个任务的方式进行描述的术语,不过并行更加注重性能
    2023-02-02
  • C语言求2的n次方多种方法总结

    C语言求2的n次方多种方法总结

    这篇文章主要给大家介绍了关于C语言求2的n次方多种方法的相关资料,求2的N次幂是一个常用的功能,文中通过代码介绍的非常详细,需要的朋友可以参考下
    2023-10-10
  • C++中memcpy和memmove的区别总结

    C++中memcpy和memmove的区别总结

    这篇文章主要介绍了C++中memcpy和memmove的区别总结,这个问题经常出现在C++的面试题目中,需要的朋友可以参考下
    2014-10-10
  • 数据结构之带头结点的单链表

    数据结构之带头结点的单链表

    单链表是一种链式存取的数据结构,用一组地址任意的存储单元存放线性表中的数据元素。链表中的数据是以结点来表示的,每个结点的构成:数据域(数据元素的映象) + 指针域(指示后继元素存储位置),元素就是存储数据的存储单元,指针就是连接每个结点的地址数据
    2023-07-07
  • C语言实现扫雷附完整代码

    C语言实现扫雷附完整代码

    本文详细讲解了C语言实现扫雷并附完整代码,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-11-11
  • C语言中实现协程案例

    C语言中实现协程案例

    这篇文章主要介绍了C语言中实现协程案例,本文通过将协程与线程和异步回调进行对比,以及具体实现案例,以下就是详细内容,需要的朋友可以参考下
    2021-07-07
  • C++ 使用PrintWindow实现窗口截图功能

    C++ 使用PrintWindow实现窗口截图功能

    这篇文章主要介绍了C++ 如何使用PrintWindow实现窗口截图功能,文中示例代码非常详细,帮助大家更好的理解和学习,感兴趣的朋友可以了解下
    2020-08-08
  • vscode 安装go第三方扩展包填坑记录的详细教程

    vscode 安装go第三方扩展包填坑记录的详细教程

    这篇文章主要介绍了vscode 安装go第三方扩展包填坑记录,本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-05-05
  • 解析C++中的for循环以及基于范围的for语句使用

    解析C++中的for循环以及基于范围的for语句使用

    这篇文章主要介绍了解析C++中的for循环以及基于范围的for语句使用,是C++入门学习中的基础知识,需要的朋友可以参考下
    2016-01-01
  • C语言面试C++字符串替换空格示例

    C语言面试C++字符串替换空格示例

    这篇文章主要介绍了C语言面试中C++字符串替换空格示例,文中给出了基本上可以拿下offer的代码,有需要的朋友可以借鉴参考下,希望大家都能早日拿到心仪的offer
    2021-09-09

最新评论