服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - C/C++ - C语言超详细梳理排序算法的使用

C语言超详细梳理排序算法的使用

2022-11-02 11:34雪芙花 C/C++

这篇文章主要介绍了C语言完成排序的实例,在C语言基本类型的排序中特别有用,下面我们一起进入文章学习更详细的内容吧,需要的朋友可以参考下

排序的概念及其运用

排序的概念

排序:所谓排序,就是使一串记录,按照其中的某个或某些关键字的大小,递增或递减的排列起来的操作。

稳定性:假定在待排序的记录序列中,存在多个具有相同的关键字的记录,若经过排序,这些记录的相对次
序保持不变,即在原序列中,r[i]=r[j],且r[i]在r[j]之前,而在排序后的序列中,r[i]仍在r[j]之前,则称这种排
序算法是稳定的;否则称为不稳定的。 内部排序:数据元素全部放在内存中的排序。

外部排序:数据元素太多不能同时放在内存中,根据排序过程的要求不能在内外存之间移动数据的排序。

排序运用

高校排名:

C语言超详细梳理排序算法的使用

接下来,我会一一介绍几种常见的排序算法

插入排序

直接插入排序

直接插入排序是一种简单的插入排序法

基本思想: 把待排序的记录按其关键码值的大小逐个插入到一个已经排好序的有序序列中,直到所有的记录插入完为止,得到一个新的有序序列

代码的实现

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
//直接插入排序
void InsertSort(int* a, int n)
{
    assert(a);//传入数组不为空指针
    int i;
    for (i = 0; i < n - 1; i++)
        
    {
        int end = i;
        int x = a[end + 1];
 
        while (end >= 0)
        {
            //升序
            if (a[end] >x)
            {
                a[end + 1] = a[end];
                end--;
            }
            else
            {
                break;
            }
        }
        a[end + 1] = x;
    }
}

希尔排序

解析

  • 希尔排序在直接排序之前,进行预排列,将某些极端数据更快的排列到数列前面,构成一个接近排列好的序列,最后再进行一次直接插入排序
  • 预排列的原理也是插入排列,只不过这里的将数组分成了gap组,分别对每一个小组进行插入排序
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
// 希尔排序
void ShellSort(int* a, int n)
{
    int gap = n;
    while (gap > 1)
    {
        gap /= 2;
        for (int i = 0; i < n - gap; i++)
        {
            int end = i;
            int x = a[end + gap];
            while (end >= 0)
            {
                if (a[end] > x)
                {
                    a[end + gap] = a[end];
                    end-=gap;
                }
                else
                    break;
            }
            a[end + gap] = x;
        }
    }
}

当gap > 1时都是预排序,目的是让数组更接近于有序。当gap == 1时,数组已经接近有序的了,这样就会很快。这样整体而言,可以达到优化的效果。我们实现后可以进行性能测试的对比

选择排序

直接选择排序

解析

每一次遍历待排序的数据元素从中选出最小(或最大)的一个元素,存放在序列的起始(或者末尾)位置,直到全部待排序的数据元素排完

代码的实现

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
// 选择排序
void SelectSort(int* a, int n)
{
    int begin = 0, end = n - 1;//记录下标
    while (begin < end)
    {
        int mini = begin;
        for (int i = begin; i <= end; i++)
        {
            //遍历找到最小数据并记录下标
            if (a[i] < a[mini])
                mini = i;
        }
        Swap(&a[begin], &a[mini]);//交换
        begin++;//缩小范围
    }
}

总结

时间复杂度:O(N^2)
空间复杂度:O(1)
稳定性:不稳定

不推荐使用

堆排序

堆排序是指利用堆(数据结构)进行选择数据的一种排序算法

基础思想:

  • 原则:

先将原数组建成堆,需要注意的是排升序要建大堆,排降序建小堆
注:以大堆为例

  • 建堆:

一个根节点与子节点数据如果不符合大堆结构,那么则对根节点数据进行向下调整,而向下调整的前提是左右子树也符合大堆结构,所以从堆尾数据的根节点位置开始向下调整建大堆

  • 排序:

大堆堆顶数据一定是待排数据中最大的,将堆顶数据与堆尾数据交换,交换后将除堆尾数据看成新堆,对现堆顶数据进行向下调整成大堆,以此循环直至排列完毕

  • 向下调整:

找到子节点中的较大数据节点比较,如果父节点数据比大子节点小则交换,直到不符合则停止向下交换,此时再次构成了一个大堆结构.

代码的实现

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
void Adjustdown(int* a, int n,int parent)
{
    int child = parent * 2 + 1;
    while (child < n)
    {
        //找到数据大的子结点
        if (child + 1 < n && a[child + 1] > a[child])
        {
            child++;
        }
        //父节点数据小于子节点就交换
        if (a[parent] < a[child])
        {
            Swap(&a[parent], &a[child]);
            //更新下标
            parent = child;
            child = parent * 2 + 1;
        }
        else//否则向下调整完毕
            break;
    }
}
 
// 堆排序(升序)建大堆
void HeapSort(int* a, int n)
{
    int i;
    //建大堆
    for (i = (n - 1 - 1) / 2; i >= 0; i--)
    {
        Adjustdown(a, n, i);
    }
    //交换调整
    for (i = n - 1; i >= 0; i--)
    {
        Swap(&a[0], &a[i]);//与当前堆尾数据交换
        Adjustdown(a, i, 0);//对交换后堆顶数据进行向下调整
    }
}

总结:

堆排序使用堆来选数,效率就高了很多。
时间复杂度:O(N*logN)
空间复杂度:O(1)
稳定性:不稳定

交换排序之冒泡排序

冒泡排序

每次遍历数组,对相邻数据进行比较,不符合排序要求则交换

代码的实现

?
1
2
3
4
5
6
7
8
9
10
11
12
13
// 冒泡排序
void BubbleSort(int* a, int n)
{
    int i, j;
    for (i = 0; i < n - 1; i++)//趟数
    {
        for (j = 0; j < n - 1 - i; j++)//比较次数
        {
            if (a[j] > a[j + 1])//满足条件
                Swap(&a[j], &a[j + 1]);//交换
        }
    }
}

总结

排序的第一篇就讲到这里了,下一篇还会讲快速排序和归并排序,希望大家多多支持!!

到此这篇关于C语言超详细梳理排序算法的使用的文章就介绍到这了,更多相关C语言 排序算法内容请搜索服务器之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持服务器之家!

原文链接:https://blog.csdn.net/yin_ming_hui/article/details/123744576

延伸 · 阅读

精彩推荐
  • C/C++C++11中使用using的方法

    C++11中使用using的方法

    在 C++ 中 using 用于声明命名空间,使用命名空间也可以防止命名冲突。这篇文章主要介绍了C++11中使用using的方法,需要的朋友可以参考下...

    冰糖葫芦很乖11162022-01-11
  • C/C++实例讲解C语言编程中的结构体对齐

    实例讲解C语言编程中的结构体对齐

    这篇文章主要介绍了C语言编程中的结构体对齐,值得注意的是一些结构体对齐的例子在不同编译器下结果可能会不同,需要的朋友可以参考下...

    阿凡卢10832021-03-30
  • C/C++c++动态内存管理与智能指针的相关知识点

    c++动态内存管理与智能指针的相关知识点

    为了更容易同时也更安全地使用动态内存,新的标准库提供了两种智能指针(smart pointer)类型来管理对象,下面这篇文章主要给大家介绍了关于c++动态内存管...

    萌之上9152022-10-11
  • C/C++C++中的Lambda函数详解

    C++中的Lambda函数详解

    大家好,本篇文章主要讲的是C++中的Lambda函数详解,感兴趣的同学赶快来看一看吧,对你有帮助的话记得收藏一下,方便下次浏览...

    求知者先4482022-07-28
  • C/C++C++ OpenCV实现银行卡号识别功能

    C++ OpenCV实现银行卡号识别功能

    这篇文章主要介绍了如何使用OpenCV C++实现银行卡号识别功能,文中的示例代码讲解详细,对我们学习OpenCV有一定帮助,需要的可以参考一下...

    Zero___Chen3462022-09-01
  • C/C++C++连接mysql数据库的两种方法小结

    C++连接mysql数据库的两种方法小结

    这篇文章主要介绍了C++连接mysql数据库的两种方法小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们...

    bmjhappy11522021-08-31
  • C/C++C语言 用while循环求和的平均值操作

    C语言 用while循环求和的平均值操作

    这篇文章主要介绍了C语言 用while循环求和的平均值操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...

    qwq hui qwq6772021-10-11
  • C/C++详解c++中的异常

    详解c++中的异常

    程序在运行过程中,有对也就有错,正确那么就不用说了,但是如果错误,那么我们如何快速的定位到错误的位置,以及知道发生了什么错误。当一个函数...

    lsgxeva6772021-11-14