各类排序详解

前言

本篇博客将为大家介绍各类排序算法,大家知道,在我们生活中,排序其实是一件很重要的事,我们在网上购物,需要根据不同的需求进行排序,异或是我们在高考完报志愿时,需要看看院校的排名,等等这些都离不开排序;今天我们就来看计算机语言中几类常见的排序算法,如果你对本文感兴趣,欢迎留言交流,下面进入正文部分。

Comparison Sorting Visualization

大家可以通过这个网址来查看各种排序的动图,以便于大家理解。

1.直接插入排序

这个顾名思义,就是将数据取出来比较,然后再插入合适的位置以达到有序的效果;这个就类似于我们玩的扑克牌,当我们摸到拍时,我们需要按照扑克牌的规则对其进行有效的排序,这里其实就是运用了插入排序的思想。

上面介绍了什么是插入排序,下面大家来看看具体代码。

void InsertSort(int* a, int n)
{int i = 0;for (i = 0; i < n - 1; i++)//这里循环的结束条件需要注意,我们最后一组是a[n-1]=a[n-2]{int end = i;//假设[0,end]有序,现在将end+1位置的值插入到[0,end],保持有序int tmp = a[end + 1];//这里需要先保存end+1位置的值,否则挪动的时候就会覆盖掉后面的数据while (end >= 0){if (a[end] > tmp){a[end + 1] = a[end];end--;}else{break;}}a[end + 1] = tmp;}
}

直接插入排序的特性总结:

1. 元素集合越接近有序,直接插入排序算法的时间效率越高

2. 时间复杂度:O(N^2)(最坏情况),O(N)(最好情况)

3. 空间复杂度:O(1),它是一种稳定的排序算法

4. 稳定性:稳定

插入排序比起我们之前学过的冒泡排序效率是要高不少的,在实践中我们会用到插入排序,所以大家需要掌握插入排序的写法。

2. 希尔排序

希尔排序是一种比较复杂的排序方法,但是它的效率是比较高的,可以说它是选择排序的一种优化,所以我们必须先理解好插入排序;

希尔排序法又称缩小增量法。希尔排序法的基本思想是:先进行预排序,然后再进行插入排序。

预排序阶段,我们会先定义一个gap,表示一次跳过的间隔,这里其实大家可以根据插入排序去理解gap,当gap等于1时,其实就变成了插入排序。

这里大家需要明确:

gap越大,大数可以越快跳到前面,小数可以越快跳到后面,但是结果越不接近有序;

gap越小,则反之。所以我们需要让gap成为一个变量。

#include<stdio.h>
void ShellSort(int* a,int n)
{int gap = n;while (gap > 1){gap = gap / 3 + 1;//这里+1是为了让gap最终等于1//当gap>1时,执行的是预排序//当gap==1时,执行的就是插入排序for (int i = 0; i < n - gap; i++){int end = i;int tmp = a[end + gap];while (end >= 0){if (tmp < a[end]){a[end + gap] = a[end];end -= gap;}else{break;}}a[end + gap] = tmp;}}
}

这里大家注意与插入排序进行对比,代码的逻辑和插入排序是类似的;这里还有一点需要大家注意,就是希尔排序的时间复杂度:O(n^1.3),这个结果大家记忆即可。

3. 选择排序

这个排序属于比较简单的一种排序,基本思想就是找出最大的放后面,找出最小的放前面,然后重复上述过程即可,从这里大家其实就可以感受到这个算法的效率是比较低的,在实践中我们一般是不用它来进行排序的。

void SelectSort(int* a, int n)
{int begin = 0;int end = n - 1;while (begin < end){int mini = begin;int maxi = begin;for (int i = begin + 1; i <= end; i++){if (a[i] > a[maxi]){maxi = i;}if (a[i] < a[mini]){mini = i;}}Swap(&a[begin], &a[mini]);if (maxi == mini){maxi = mini;}Swap(&a[end], &a[maxi]);begin++;end--;}
}

直接选择排序的特性总结:

1. 直接选择排序思考非常好理解,但是效率不是很好。实际中很少使用

2. 时间复杂度:O(N^2)

3. 空间复杂度:O(1)

4. 稳定性:不稳定

4.快速排序

说到快速排序,这里有三种方法:hoare、挖坑法,前后指针法;我主要为大家介绍前后指针的方法,因为这个方法相对于其他两个好理解一些。

前后指针,顾名思义有两个指针,具体怎么工作呢:刚cur位置的值小于key时,那么prev++,然后将prev位置的值和cur位置的值互换,然后cur指针继续往后走;当cur位置的值大于key时,cur直接往后走;一直到cur越界为止。本质上实现key左边都是比key小的,key右边都是比key大的。

void Swap(int* p1, int* p2)
{int tmp = *p1;*p1 = *p2;*p2 = tmp;
}int PartSort(int* a, int left, int right)
{int keyi = left;int prev = left;int cur = prev + 1;while (cur <= right){if (a[cur] < a[keyi] && ++prev != cur){Swap(&a[cur], &a[prev]);}cur++;}Swap(&a[keyi], &a[prev]);return prev;
}
void QuickSort(int* a, int left, int right)
{if (left >= right){return;}int keyi = PartSort(a, left, right);QuickSort(a, left, keyi - 1);QuickSort(a, keyi + 1, right);
}

这里大家看快排的代码,这里运用了递归的思想,在单趟排序中我们得到的就是key的位置,然后在不同的区间进行快排;这里运用递归,很快可以实现快排,但是递归还是存在一定的缺陷,当深度过深时,递归可能会引发栈溢出的问题,所以我们能不能不用递归来实现快排呢?答案当然是可以的。

如果我们想写非递归的快排,那么我们就需要用到栈这个数据结构,本质上是将存在栈帧中的区间放在了栈这个数据结构中,来模拟实现递归的过程;

void Swap(int* p1, int* p2)
{int tmp = *p1;*p1 = *p2;*p2 = tmp;
}
int PartSort2(int* a, int left, int right)
{int keyi = left;int prev = left;int cur = prev + 1;while (cur <= right){if (a[cur] > a[keyi] && ++prev != cur)Swap(&a[prev], &a[cur]);cur++;}Swap(&a[prev], &a[keyi]);return prev;
}
void QuickSortNonR(int* a, int left, int right)
{ST st;STInit(&st);//入两个整数,先入右再入左,形成一个区间STPush(&st, right);STPush(&st, left);while (!STEmpty(&st)){int begin = STTop(&st);//这里出栈就是先出左再出右STPop(&st);int end = STTop(&st);STPop(&st);int keyi = PartSort2(a, begin, end);if (keyi + 1 < end){STPush(&st, end);STPush(&st, keyi+1);}if (begin < keyi - 1){STPush(&st, keyi-1);STPush(&st, begin);}}STDestroy(&st);
}

这里大家来看非递归的代码,循环每走一次,就取出栈顶区间,然后还是先入右再入左,如此循环下去,就模拟了递归的过程。 

快速排序的特性总结:

1. 快速排序整体的综合性能和使用场景都是比较好的,所以才敢叫快速排序

2. 时间复杂度:O(N*logN)

3. 空间复杂度:O(logN)

4. 稳定性:不稳定

5. 堆排序

堆排序(Heapsort)是指利用堆积树(堆)这种数据结构所设计的一种排序算法,它是选择排序的一种。它是通过堆来进行选择数据。

需要注意的是排升序要建大堆,排降序建小堆。

堆排序之所以叫做堆排是因为需要结合堆中的向上调整和向下调整两个算法,具体大家可以看下面代码或者去二叉树那篇文章中查看。

void Swap(int* p1, int* p2)
{int tmp = *p1;*p1 = *p2;*p2 = tmp;
}
void AdjustUp(int* a, int child)
{int parent = (child - 1) / 2;while (child >= 0){if (a[child] > a[parent]){Swap(&a[child], &a[parent]);child = parent;parent = (child - 1) / 2;}else{break;}}
}
void AdjustDown(int* a, int n, int parent)
{//这里先假设左孩子小int child = parent * 2 + 1;while (child < n)//child>=n时,说明已经调整到叶节点了{//找出小的那个孩子if (a[child + 1] > a[child] && child + 1 < n){child++;}if (a[child] > a[parent]){Swap(&a[child], &a[parent]);parent = child;child = parent * 2 + 1;}else{break;}}
}
void HeapSort(int* a, int n)
{int i = 0;for (i = 1; i < n; i++){AdjustUp(a, i);}int end = n - 1;while (end > 0){Swap(&a[0], &a[end]);AdjustDown(a, end, 0);end--;}
}

这里我举的例子是建大堆排升序,升降序可以在向下调整和向上调整中进行切换。

堆排的工作原理在于我们要先根据需要建堆,这里就用到向上调整的算法帮助我们建成大堆或小堆;建好堆后,我们需要将首尾位置的数据进行交换,然后让根部位置的数据向下调整,这里拿建大堆排升序来说,我们第一次交换将最大的数放到了最后,然后第二次将次大的数放到倒数第二个位置,以此类推,到最后我们就可以得到一组升序的数据;排降序是同理。 

上面所介绍的是堆排的第一种写法,这种写法是比较容易理解的;但是,我们还有另一种写法,可以在其基础上再提高效率——向下调整建堆。这里大家需要明确一个问题向下调整算法有一个前提:左右子树必须是一个堆,才能调整。所以我们必须保证左右子树都是堆才可以,然而这样显然不太现实,于是我们就采取一种倒着建堆的方式,我们从倒数第一个非叶子节点开始调,一直到根节点,这样也能实现同样的建堆效果。

这里大家还需要注意一点:无论哪种写法,时间复杂度都为:O(NlogN)。但是实际上还是第二种方法能快一些,它们时间复杂度一样,只是说它们属于同一个量级,但是具体详细地来看,还是第二种快,建议大家堆排就写下面这种。

void AdjustDown(int* a, int n, int parent)
{//这里先假设左孩子小int child = parent * 2 + 1;while (child < n)//child>=n时,说明已经调整到叶节点了{//找出小的那个孩子if (a[child + 1] < a[child] && child + 1 < n){child++;}if (a[child] < a[parent]){Swap(&a[child], &a[parent]);parent = child;child = parent * 2 + 1;}else{break;}}
}
void HeapSort(int* a, int n)
{int i = 0;for (i = (n-2)/2; i >=0; i--){AdjustDown(a, n, i);}int end = n - 1;while (end > 0){Swap(&a[0], &a[end]);AdjustDown(a, end, 0);end--;}
}

堆排序的特性总结:

1. 堆排序使用堆来选数,效率就高了很多。

2. 时间复杂度:O(N*logN)

3. 空间复杂度:O(1)

4. 稳定性:不稳定

堆排序是一种效率很高的排序方法,尤其是在数据比较多的情况下,它的运行效率和快排是一个档次的,所以这个排序是具有实际意义的。

6. 归并排序

大家可以直接来看这张图,展现了归并排序的核心思想,我们要采用分治法,保证两边区间都有序后进行归并;

void _MergeSort(int* a, int* tmp, int begin, int end)
{if (begin == end)return;int mid = (begin + end) / 2;//递归使左右区间都有序_MergeSort(a, tmp, begin, mid);_MergeSort(a, tmp, mid+1, end);//进行归并int begin1 = begin, end1 = mid;int begin2 = mid+1, end2 = end;int i = begin;while (begin1 <= end1 && begin2 <= end2){if (a[begin1] < a[begin2]){tmp[i++] = a[begin1++];}else{tmp[i++] = a[begin2++];}}while (begin1 <= end1){tmp[i++] = a[begin1++];}while (begin2 <= end2){tmp[i++] = a[begin2++];}memcpy(a + begin, tmp + begin, (end - begin + 1) * sizeof(int));
}
void MergeSort(int* a, int n)
{int* tmp = (int*)malloc(sizeof(int) * n);if (tmp == NULL){perror("malloc fail");return;}_MergeSort(a, tmp, 0, n - 1);free(tmp);tmp = NULL;
}

 上面就是归并排序的代码,其实大家可以结合二叉树中的后序遍历来进行理解,这要提醒一下大家,在我们分割区间的时候,要按照上面的方式进行分割,不能分割成[begin,mid-1]和[mid,end],这样会导致程序陷入死循环。

上面我们学习快排的时候,介绍了两种方法,一种是运用递归,一种是运用非递归;那么在这里亦有异曲同工之妙,归并排序也是可以用非递归来实现的。

void MergeSortNonR(int* a, int n)
{int* tmp = (int*)malloc(sizeof(int) * n);if (tmp == NULL){perror("malloc fail");exit(1);}//gap为每组归并数据的个数int gap = 1;int i = 0;while (gap < n){for (i = 0; i < n; i += 2 * gap){//进行归并int begin1 = i, end1 = i + gap - 1;int begin2 = i + gap, end2 = i + 2 * gap - 1;if (begin2 >= n){break;}if (end2 >= n){end2 = n - 1;}int j = i;while (begin1 <= end1 && begin2 <= end2){if (a[begin1] < a[begin2]){tmp[j++] = a[begin1++];}else{tmp[j++] = a[begin2++];}}while (begin1 <= end1){tmp[j++] = a[begin1++];}while (begin2 <= end2){tmp[j++] = a[begin2++];}memcpy(a + i, tmp + i, (end2 - i + 1) * sizeof(int));}gap *= 2;}free(tmp);tmp = NULL;
}

这里大家来看一下非递归的归并排序,核心思想就是一组一组归并,这里大家要注意我们需要归并一段拷贝一段,区别于上面递归版的归并。

归并排序的特性总结:

1. 归并的缺点在于需要O(N)的空间复杂度,归并排序的思考更多的是解决在磁盘中的外排序问题。

2. 时间复杂度:O(N*logN)

3. 空间复杂度:O(N)

4. 稳定性:稳定

这里大家可以发现,归并排序也是一种效率很高的排序方法,它的效率和堆排快排是一个档次的,所以具有一定实践意义。

7. 冒泡排序

关于冒泡排序,这里就不赘述了,再前面C语言的文章介绍循环的文章中有,大家有需要的可以自行去我的主页查看。为什么里不详细介绍冒泡呢?原因有二:

其一,冒泡排序比较简单,属于交换排序的一种,如果是初学者可以学习一下冒泡排序,感受一下循环的嵌套。

其二,冒泡排序确实是上不了台面,因为它太慢了,和其他几个不是一个量级的,所以没啥实际意义,只有一定的教学意义。

8. 计数排序

void CountSort(int* a, int n)
{int min = a[0];int max = a[0];int i = 0;for (i = 0; i < n; i++){if (a[i] < min){min = a[i];}if (a[i] > max){max = a[i];}}int range = max - min + 1;int* count = (int*)calloc(range, sizeof(int));if (count == NULL){perror("calloc fail");exit(1);}//统计次数for (i = 0; i < n; i++){count[a[i] - min]++;}//排序int j = 0;for (i = 0; i < range; i++){while (count[i]--){a[j++] = i + min;}}free(count);count = NULL;
}

这里需要为大家强调几点,计数排序只适用于整数,适合范围集中的数据;但是计数排序在特定情况下的效率是很高的,甚至比堆排还要快,所以具有实践意义,我们可以在日常中使用它来进行排序。

 9. 各类排序算法分析

这里主要就是来总结一下各类排序的时间复杂度、空间复杂度、稳定性等因素;首先大家需要明确一下稳定性的一个概念:相同的值相对位置不变。满足这个要求才可以算得上稳定。

10.总结

本篇博客为大家介绍了各种排序算法,其中有一些算法的实践意义不大,有些是可以在实践中进行运用的,大家需要理解每一种排序的思路,并且掌握对应的代码,了解每种排序的特点;排序这块儿内容还是比较重要的,后面我们在面试中,排序是一个常考的点,所以大家需要重点掌握,最后,希望本篇博客可以为大家带来帮助,感谢阅读!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/443992.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

qt QGraphicsItem详解

一、概述 QGraphicsItem是Qt框架中图形视图框架&#xff08;Graphics View Framework&#xff09;的一个核心组件&#xff0c;它是用于表示2D图形元素的基类。 它支持的功能包括&#xff1a; 设置和获取图形项的位置和尺寸。控制图形项的外观&#xff0c;如颜色、笔刷、边框…

京东web 京东e卡绑定 第二部分分析

声明 本文章中所有内容仅供学习交流使用&#xff0c;不用于其他任何目的&#xff0c;抓包内容、敏感网址、数据接口等均已做脱敏处理&#xff0c;严禁用于商业用途和非法用途&#xff0c;否则由此产生的一切后果均与作者无关&#xff01; 有相关问题请第一时间头像私信联系我删…

请求参数中字符串的+变成了空格

前端请求 后端接收到的结果 在URL中&#xff0c;某些字符&#xff08;包括空格、、&、? 等&#xff09;需要被编码。具体而言&#xff0c;在URL中&#xff0c;空格通常被编码为 或 %20。因此&#xff0c;如果你在请求参数中使用 &#xff0c;它会被解释为一个空格。 如果…

2024重生之回溯数据结构与算法系列学习(12)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丟脸好嘛?】

欢迎各位彦祖与热巴畅游本人专栏与博客 你的三连是我最大的动力 以下图片仅代表专栏特色 [点击箭头指向的专栏名即可闪现] 专栏跑道一 ➡️ MYSQL REDIS Advance operation 专栏跑道二➡️ 24 Network Security -LJS ​ ​ ​ 专栏跑道三 ➡️HCIP&#xff1b;H3C-SE;CCIP——…

智能边缘计算 | 项目快速部署指南

在数字化浪潮的推动下&#xff0c;边缘计算与人工智能的深度融合正在成为推动智能社会发展的新动力。 边缘计算通过将数据处理和分析任务从中心服务器转移到更接近数据源的端侧&#xff0c;从而显著降低数据传输延迟&#xff0c;提高了响应速度和安全隐私性。在人工智能的加持…

python的特殊方法——魔术方法

前言 __init__(self[]) ​编辑 __call__(self [, ...]) __getitem__(self, key) __len__(self) __repr__(self) / __str__(self) __add__(self, other) __radd__(self, other) 参考文献 前言 官方定义好的&#xff0c;以两个下划线开头且以两个下划线结尾来命名的方法…

在QT中将Widget提升为自定义的Widget后,无法设置Widget的背景颜色问题解决方法

一、问题 在Qt中将QWidget组件提升为自定义的QWidget后&#xff0c;Widget设置的样式失效&#xff0c;例如设置背景颜色为白色失效。 二、解决方法 将已经提升的QWidget实例对象&#xff0c;脱离父窗体的样式&#xff0c;然后再重新设置自己的样式。

[ComfyUI]太赞了!阿里妈妈发布升级版 Flux 图像修复模型,更强细节生成,更高融合度以及更大分辨率支持

小伙伴们还记得我们之前介绍的阿里妈妈发布的 Flux 的 ControlNet 图像修复模型不&#xff0c;之前发布的是 Alpha 早期测试版本&#xff0c;说实话和 Flux 原生的重绘其实差距不大&#xff0c;有些方面甚至还是原生的效果更好。 但是现在&#xff0c;Alpha 的升级版本 Beta 版…

Stable Diffusion绘画 | 签名、字体、Logo设计

第1步&#xff0c;使用 PS&#xff08;小白推荐使用 可画&#xff09;准备一个 512*768 的签名、字体、Logo图片&#xff1a; 第2步&#xff0c;来到模型网站&#xff0c;搜索&#x1f50d;关键词“电商”&#xff0c;找到一款喜欢的 LoRA&#xff1a; 第3步&#xff0c;选择一…

4.STM32-中断

STM32-中断 需求&#xff1a;红灯每两秒进行闪烁&#xff0c;按键key1控制绿灯亮灭 简单的程序代码无法满足要求 如何让STM32既能执行HAL_DELAY这种耗时的任务&#xff0c;同时又能快速响应按键按下这种突发情况呢 设置中断步骤 1.接入中断 将KEY1输入模式由原先的GPIO_In…

布隆过滤器基本原理与使用

目录 1.引言 2.基本定义 3.基本原理 4.实现方法 5.布隆过滤器的优缺点 6.哈希冲突和误判问题 7.大规模数据集Redis中布隆过滤器的性能优化 8.应用场景举例 1.引言 在互联网应用中&#xff0c;随着用户基数和交互数据的爆炸性增长&#xff0c;如何高效地处理点赞、签到、…

vue出现Component name “Politic“ should always be multi-word错误

效果 原因 组件名不能为单个单词&#xff0c;怕和html标签混淆 解决方法 1.选择多个单词区分 2.修改package.json里的rules规则&#xff0c;忽略文件命名校验

跨平台游戏的特点

跨平台游戏已成为视频游戏行业的主要趋势&#xff0c;这是由于对各种设备和操作系统之间无缝游戏的需求日益增长。这种方法允许玩家在多个平台&#xff08;如游戏机、PC和移动设备&#xff09;上享受他们最喜欢的游戏&#xff0c;同时保持相同的体验和进度。随着越来越多的开发…

【2024年最新】基于springboot+mysql就业信息管理系统

技术摘要 技术框架&#xff1a;以springboot作为框架&#xff0c;业务模式&#xff1a;B/S模式数据库&#xff1a;MySql作为后台运行的数据库服务器&#xff1a;使用Tomcat用为系统的服务器 系统展示 系统实现功能 本次实现一个就业信息管理系统&#xff0c;通过这个系统能够满…

【北京迅为】《STM32MP157开发板嵌入式开发指南》-第二十二章 安装VMware Tool 工具

iTOP-STM32MP157开发板采用ST推出的双核cortex-A7单核cortex-M4异构处理器&#xff0c;既可用Linux、又可以用于STM32单片机开发。开发板采用核心板底板结构&#xff0c;主频650M、1G内存、8G存储&#xff0c;核心板采用工业级板对板连接器&#xff0c;高可靠&#xff0c;牢固耐…

ssrf学习(ctfhub靶场)

ssrf练习 目录 ssrf漏洞 漏洞形成原理&#xff08;来自网络&#xff09; 寻找ssrf漏洞&#xff0c; 靶场题目 第一题&#xff08;url探测网站下文件&#xff09; 第二关&#xff08;使用伪协议&#xff09; 关于http和file协议的理解 file协议 http协议 第三关&…

猫头虎分享已解决Bug || Error: ERESOLVE unable to resolve dependency tree 解决方案

&#x1f42f; 猫头虎分享已解决Bug || Error: ERESOLVE unable to resolve dependency tree 解决方案 摘要 在前端开发中&#xff0c;尤其是使用 Node.js 和 npm 管理依赖时&#xff0c;ERESOLVE unable to resolve dependency tree 错误是很多开发者遇到的常见问题。这个 Bu…

jQuery 用户登录页面非空校验与登录测试

文章目录 实战介绍准备工作创建网页导入样式表和jQuery库编写页面代码编写脚本代码创建成功页面浏览网页和测试结束语 实战介绍 大家好&#xff0c;今天我们将一起学习如何使用jQuery来为用户登录页面进行非空校验和登录测试。通过这个实战项目&#xff0c;你将学会如何通过jQ…

新版 Notepad++ 下载与安装教程

一、软件准备&#xff1a;麻烦点我 二、双击下载好的 notepad 软件进行安装&#xff0c;选择 “简体中文”。 三、默认 “下一步” 安装。 四、单击 “我接受” 按钮。 五、自定义安装位置&#xff0c;个人建议安装在 D 盘。 六、选择组件&#xff0c;默认 “下一步”。 七、勾…

使用Diskgenius系统迁移

使用Diskgenius系统迁移 1、使用系统迁移2、注意点3、新备份的系统盘装在电脑上可能出现盘符错乱导致开机不进入桌面情况 1、使用系统迁移 参考视频&#xff1a; DiskGenius无损系统迁移&#xff0c;换硬盘无需重装系统和软件 2、注意点 1&#xff09;新的硬盘里面的所有资料…