排序算法合集(2)

news2025/1/15 13:08:09

前言:

上一篇文章,我们已经讲完了冒泡排序,选择排序,插入排序和希尔排序。

那么我们今天来讲一下堆排序,快速排序和归并排序吧~

堆排序(Heap Sort)

堆排序是一种基于完全二叉树的排序算法。堆排序的基本思想是先将待排序的数据构建成一个大根堆或小根堆,然后将堆顶元素与堆底元素交换位置,然后将堆的大小减一,再调整堆使其重新成为一个大根堆或小根堆,重复上述步骤直到堆的大小为1为止。

构建大根堆或小根堆的过程可以通过自下而上或自上而下两种方式进行。自下而上的方式,从最后一个非叶子节点开始,依次将每个节点作为根节点进行调整,使其成为一个子树的大根堆或小根堆。自上而下的方式,则是从根节点开始,依次将左右子树调整为大根堆或小根堆,然后再将整个树调整为一个大根堆或小根堆。

堆排序的优点是不需要额外的辅助空间,而且可以在最坏情况下保证 O(nlogn) 的时间复杂度。缺点是不稳定,无法保证相同元素的相对位置不变。

来看下面的动图:

看完动图之后,直接来看代码吧~

void heap_sort(int arr[], int n);
void heapify(int arr[], int n, int i);

void heap_sort(int arr[], int n) {
    // 构建大根堆
    for (int i = n / 2 - 1; i >= 0; i--)
        heapify(arr, n, i);
    
    // 进行堆排序
    for (int i = n - 1; i > 0; i--) {
        int temp = arr[0];
        arr[0] = arr[i];
        arr[i] = temp;
        heapify(arr, i, 0);
    }
}

void heapify(int arr[], int n, int i) {
    int largest = i;
    int left = 2 * i + 1;
    int right = 2 * i + 2;
    
    if (left < n && arr[left] > arr[largest])
        largest = left;
    
    if (right < n && arr[right] > arr[largest])
        largest = right;
    
    if (largest != i) {
        int temp = arr[i];
        arr[i] = arr[largest];
        arr[largest] = temp;
        heapify(arr, n, largest);
    }
}

 堆的数据结构和性质是堆排序的灵魂,是堆排序能够高效实现的关键。

快速排序(Quick Sort)(递归实现)

快速排序是常用的一种基于分治思想的排序算法。

快速排序的基本思想是选定一个基准元素(pivot),将待排序的数据分为两个子序列,其中一个子序列中的所有元素都比基准元素小,另一个子序列中的所有元素都比基准元素大。然后分别对这两个子序列递归地进行快速排序,最终将整个序列排序。

快速排序的关键在于如何选取基准元素。一般来说,可以选择序列的第一个元素、最后一个元素、中间元素或随机元素作为基准元素。

快速排序也有三种版本,分别为hoare版本,挖坑法,前后指针法

那么我们先来看hoare

hoare

看到上面的动图了吗?

我们一开始把左边小人站的位置的值当作基准值,我们的右边小人就会开始往左走,直到找到了比我们基准值还要小的数停下,之后我们的左边小人才会开始向右走,直到找到一个比基准值还要大的值,之后两个小人所站的两个值互换即可,依次进行下去,两个小人就会在某个地方会合,那么我们会把基准值和两个小人会合的值互换位置,这就完成了一次快速排序,之后将这次基准值的左右侧分别进行快速排序进行递归就可以完成这整一套的快速排序啦

那么我们就看代码吧~

void QuickSort1(int* a, int left, int right)
{
	if (left >= right)
		return;

	int begin = left, end = right;

	 随机选key
	/*int randi = left + (rand() % (right - left));
	Swap(&a[left], &a[randi]);*/

	// 三数取中
	int midi = GetMidNumi(a, left, right);
	if(midi != left)
		Swap(&a[midi], &a[left]);

	int keyi = left;
	while (left < right)
	{
		// 右边找小
		while (left < right && a[right] >= a[keyi])
			--right;

		// 左边找大
		while (left < right && a[left] <= a[keyi])
			++left;

		Swap(&a[left], &a[right]);
	}

	Swap(&a[keyi], &a[left]);
	keyi = left;

	// [begin, keyi-1] keyi [keyi+1, end] 
	// 递归
	QuickSort1(a, begin, keyi - 1);
	QuickSort1(a, keyi+1, end);
}

 我们的递归结束条件就是区间的left>=right。

挖坑法

挖坑法快速排序是快速排序的一种变体,相比于传统的交换元素法,它的优势在于减少了交换操作的次数,从而提高了排序的效率。

在传统的快速排序中,每当找到一个需要交换的元素时,需要进行一次交换操作。而在挖坑法快速排序中,只需要将需要交换的元素的值拷贝到一个临时变量中,然后用这个值去填补基准元素的坑位,从而达到交换元素的目的。这样,就避免了不必要的交换操作,提高了排序的效率。

当然,除了提高了排序的效率,我认为,挖坑法比传统的Hoare更好理解。

那么老样子,还是先来看动图吧~

看完了动图,我们马上就可以写出它的代码

void QuickSort2(int* a, int left, int right)
{
	if (left >= right)
		return;

	int begin = left, end = right;

	// 三数取中
	int midi = GetMidNumi(a, left, right);
	if (midi != left)
		Swap(&a[midi], &a[left]);

	
	int key = a[left];
	int hole = left;
	while (left < right)
	{
		// 右边找小
		while (left < right && a[right] >= key)
			--right;

		a[hole] = a[right];
		hole = right;

		// 左边找大
		while (left < right && a[left] <= key)
			++left;

		a[hole] = a[left];
		hole = left;
	}

	a[hole] = key;

	// [begin, hole-1] hole [hole+1, end] 
	// 递归
	QuickSort2(a, begin, hole - 1);
	QuickSort2(a, hole + 1, end);
}

前后指针法

前后指针法是快速排序的一种常用实现方式,其基本思想是通过两个指针从序列的两端开始向中间移动,找到需要交换的元素并进行交换,最终完成排序。

cur一直向前遍历,只有cur位置上的值比基准值小时,prev向前遍历。当cur和prev之间有差距,说明两者之间都为比基准值大的元素,交换cur和prev上的元素。最后将基准值和prev最后停下来位置上的元素进行交换。

来看动图

看完动图就来写代码吧~ 

void quickSort3(int arr[], int left, int right) {
    int i, j, pivot, temp;
    if (left < right) {
        pivot = arr[left];
        i = left;
        j = right;
        while (i < j) {
            while (i < j && arr[j] > pivot)
                j--;
            while (i < j && arr[i] <= pivot)
                i++;
            if (i < j) {
                temp = arr[i];
                arr[i] = arr[j];
                arr[j] = temp;
            }
        }
        arr[left] = arr[i];
        arr[i] = pivot;
        quickSort3(arr, left, i - 1);
        quickSort3(arr, i + 1, right);
    }
}

以上均为递归实现的快速排序~

那么我们还要来学习非递归的快速排序

快速排序(非递归)

有人可能会问了,正常情况下怎么样才能实现递归呢?

其实仔细看这个快速排序,如果两边分配的均匀的话,是不是排序有点像分解一棵二叉树?

我们就需要通过一个介质来保存每一层递归的信息,比如左右边界,还有递归的深度,那么模拟实现非递归的介质就是------栈~

void quickSort(int arr[], int left, int right) {
    int stack[128];
    int top = -1;
    stack[++top] = left;
    stack[++top] = right;
    while (top >= 0) {
        right = stack[top--];
        left = stack[top--];
        int i = left;
        int j = right;
        int pivot = arr[left];
        while (i < j) {
            while (i < j && arr[j] >= pivot)
                j--;
            if (i < j)
                arr[i++] = arr[j];
            while (i < j && arr[i] <= pivot)
                i++;
            if (i < j)
                arr[j--] = arr[i];
        }
        arr[i] = pivot;
        if (left < i - 1) {
            stack[++top] = left;
            stack[++top] = i - 1;
        }
        if (i + 1 < right) {
            stack[++top] = i + 1;
            stack[++top] = right;
        }
    }
}

int main() {
    int arr[] = {5, 2, 6, 3, 1, 4};
    int n = sizeof(arr) / sizeof(arr[0]);
    quickSort(arr, 0, n - 1);
    for (int i = 0; i < n; i++)
        printf("%d ", arr[i]);
    printf("\n");
    return 0;
}

也许有同学发现了上面的一些特殊代码,例如:三数取中。

这就要谈谈实际出现的特殊情况了,如果一开始我们的序列就是有序的或者是基本有序的,然后我们又选出的是序列的开头的值为基准值,这个基准值很有可能是整个序列的最大值或最小值,我们所说的形似二叉树的样子就不太像了,他会分出两个子序列,一个子序列会特别的短,一个子序列会特别的长,那么我们递归排序要处理的数据就会十分的庞大,也就是说我们的快速排序的时间复杂度也会随之变大,变成了O(N^2),那么为了解决这个问题,我们有三种优化策略。

分别为:三路归并优化,三数取中优化,还有随机抽取基准值优化

三路归并

在普通的快速排序中,将序列分为小于和大于基准元素的两部分,而在三路归并优化中,将序列分为小于、等于和大于基准元素三部分,可以有效地处理重复元素。具体来说,将序列分成三部分后,对小于、等于和大于基准元素的部分分别进行递归排序,最后将三部分合并成一个有序序列

三数取中

在选择基准元素时,如果每次都选择序列的第一个或最后一个元素,可能会导致快速排序的时间复杂度退化为O(n^2),因为如果序列本身已经有序或基本有序,每次选择的基准元素都是最大或最小值,这样分割出来的两部分序列大小差别很大,递归排序时需要处理的数据量也很大。为了避免这种情况,可以采用三数取中优化,即从序列的第一个、中间和最后一个元素中选择中间大小的元素作为基准元素,这样可以尽量保证分割出来的两部分序列大小相近,从而减少递归排序时需要处理的数据量,提高快速排序的效率。

随机抽取基准值

在选择基准元素时,也可以采用随机抽取基准值的方式,即从待排序序列中随机抽取一个元素作为基准元素,这样可以避免序列已经有序或基本有序的情况,从而提高快速排序的效率。此外,随机抽取基准值的方式也可以降低快速排序被恶意攻击的风险,比如有人故意构造一个特殊的序列,使得快速排序的时间复杂度退化到O(n^2),从而导致程序崩溃或性能下降。

归并排序(Merge Sort)(递归)

归并排序是一种基于分治思想的排序算法,其基本思想是将待排序序列分成若干个子序列,对每个子序列进行排序,然后将子序列合并成一个有序序列。将子序列合并的过程中,需要使用额外的空间来存储合并后的序列,因此归并排序的空间复杂度较高。

具体来说,归并排序可以使用递归或非递归的方式实现。在递归实现中,将待排序序列不断分成左右两个子序列,对左右两个子序列分别进行递归排序,最后将两个有序子序列合并成一个有序序列。在非递归实现中,使用循环迭代的方式对子序列进行归并排序,最终得到有序序列。

 

// 合并两个有序序列
void merge(int arr[], int left, int mid, int right, int *temp) {
    int i = left;
    int j = mid + 1;
    int k = 0;
    while (i <= mid && j <= right) {
        if (arr[i] <= arr[j])
            temp[k++] = arr[i++];
        else
            temp[k++] = arr[j++];
    }
    while (i <= mid)
        temp[k++] = arr[i++];
    while (j <= right)
        temp[k++] = arr[j++];
    for (i = 0; i < k; i++)
        arr[left + i] = temp[i];
}

// 归并排序
void mergeSort(int arr[], int left, int right, int *temp) {
    if (left < right) {
        int mid = (left + right) / 2;
        mergeSort(arr, left, mid, temp);
        mergeSort(arr, mid + 1, right, temp);
        merge(arr, left, mid, right, temp);
    }
}

一直一分为二,直到出现一个或零个时,递归结束,开始往回返,开始合并成一个有序的子序列。之后每一个子序列一直合并就会合成一个有序的序列了。.

那么我们可以用非递归实现吗?

当然是可以的,那我们还需要借助栈来实现吗?

答案是不用!

我们直接通过迭代就可以实现非递归的归并排序

// 归并排序的非递归实现
void mergeSort(int arr[], int n) {
    int *temp = (int *) malloc(n * sizeof(int));
    for (int step = 1; step < n; step *= 2) {
        for (int i = 0; i < n - step; i += 2 * step) {
            int left = i;
            int mid = i + step - 1;
            int right = i + 2 * step - 1 < n ? i + 2 * step - 1 : n - 1;
            int i1 = left, i2 = mid + 1, k = 0;
            while (i1 <= mid && i2 <= right) {
                if (arr[i1] <= arr[i2])
                    temp[k++] = arr[i1++];
                else
                    temp[k++] = arr[i2++];
            }
            while (i1 <= mid)
                temp[k++] = arr[i1++];
            while (i2 <= right)
                temp[k++] = arr[i2++];
            for (int j = 0; j < k; j++)
                arr[left + j] = temp[j];
        }
    }
    free(temp);
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

ubuntu安装php8.1

ubuntu安装php8.1 设置apt库安装在 Ubuntu 22.04|20.04|18.04 上安装 PHP 8.1 扩展安装常用的php扩展完成后输入看扩展安装情况 重启php PHP8是PHP最新的Release版本&#xff0c;在这个版本也加入了很多新的功能。 但是在Ubuntu的默认apt库里&#xff0c;找不到PHP8&#xff0c…

Android14新权限机制

Android14新权限机制 在本文中&#xff0c;我们将了解 Android14 的新权限&#xff0c;该权限允许用户仅授予对选定媒体&#xff08;照片/视频&#xff09;的访问权限&#xff0c;而不是整个库。 这个新权限允许用户在我们的应用程序运行时授予部分访问权限 mediaAndroid14 or …

Unity3D:立方体贴图

推荐&#xff1a;将 NSDT场景编辑器 加入你的3D工具链 3D工具集&#xff1a; NSDT简石数字孪生 立方体贴图 __立方体贴图__是六个正方形纹理的集合&#xff0c; 这些纹理代表环境中的反射。六个正方形组成一个包围着对象的虚构立方体的各面&#xff1b; 每个面代表沿世界轴方向…

colmap多相机重建多场景及数据库数据快速修改方法

1 colmap流程 1.1 新建项目 首先打开colmap&#xff0c;然后创建新的project&#xff0c;其中数据库目录和名称自己选定&#xff0c;注意不要将它放到图像目录下即可。然后images选择的是图像目录&#xff08;比如我这里是guangxi/section1&#xff09;&#xff0c;这个目录下…

聊天机器人简要理解

聊天机器人 身边的同事或多或少都在聊ChatGPT&#xff0c;这又使得学生时代训练模型的恐惧感开始支配我。说起聊天机器人&#xff0c;我们先熟悉一下不同对话系统的概念。 一、对话系统分类 1、单轮对话 单轮与传统的问答系统相类似&#xff0c;是智能对话系统的初级应用。…

时代变迁的缩影:从天涯BBS的沦落,到媲美EXCEL的WPS后辈的腾达

昔日王者&#xff0c;今朝黯然 “无法访问此网站” 陪伴我们青春的天涯论坛就此真的与我们相忘于天涯。 江湖路远&#xff0c;不知何日才能重逢&#xff1f; 这是第几个了&#xff1f; 那个人人自由创造的时代是否真的已经逝去&#xff1f; 近年来网易、凤凰等论坛相继关闭。…

Docker Harbor

目录 一、Docker Harbor概述 1、Harbor的优势 2、Harbor知识点 3、Docker私有仓库架构 二、Harbor构建Docker私有仓库 1、环境配置 2、案例需求 3、部署docker-compose服务 4、部署harbor服务 5、启动harbor ① 访问 ② 添加项目并填写项目名称 ③ 通过127.0.0.1来…

大江南北十三省,蓬门今始为君开,数字政府建设“实在”必行

引言&#xff1a; 推动数字中国建设是实现中国式数字化进程、抢占未来发展制高点的关键。为全面推动数字化转型在政府领域的纵深发展&#xff0c;各地领导不断进行数字化探索。其中&#xff0c;数字员工凭借流动的数据、流畅的体验&#xff0c;百姓少跑腿、数据多跑路特点&…

零基础也能轻松搭建Python环境和pycharm

目录&#xff1a;导读 引言 python环境安装 pycharm安装 设置 字体的大小&#xff1a; 关闭警告&#xff1a; 代码风格&#xff1a; 代码层级&#xff1a; ide与版本控制的整合&#xff1a; 插件安装&#xff1a;&#xff08;3个&#xff09; 介绍&#xff1a; 控制台&…

HJTB-9222跳闸闭锁继电器导轨安装 约瑟JOSEF

名称&#xff1a;跳闸闭锁继电器品牌&#xff1a;JOSEF约瑟型号&#xff1a;HJTB-9222额定电压&#xff1a;110、220V触点容量&#xff1a;250V/5A动作时间&#xff1a;≤10mS HJTB系列跳闸闭锁继电器 系列型号 HJTB-9222跳闸闭锁继电器 概述和应用 HJTB系列跳闸闭锁继电器…

NLP模型(四)——seq2seq模型与Attention机制介绍

文章目录 1. seq2seq模型介绍2. 模型计算2.1 LSTM结构单元2.2 seq2seq计算流程 3. Attention机制3.1 引入Attention3.2 计算全局对齐权重 4. seq2seq加入Attention机制5. 模型理解6. 模型细节6.1 解码器结构6.2 加入信息方式 参考文章 1. seq2seq模型介绍 传统的RNN只能处理 1…

诺派克ROPEX热封控制器维修RES-407/RES-406

德国ROPEX热封控制器维修型号包括&#xff1a;RES-401&#xff0c;RES-402&#xff0c;RES-403&#xff0c;RES-406&#xff0c;RES-407&#xff0c;RES-408&#xff0c;RES-409&#xff0c;RES-420&#xff0c;RES-440&#xff0c;MSW-2&#xff0c;PEX-W3&#xff0c;PEX-W4&…

Android 源码浅析:Leakcanary 内存泄漏检测的好帮手

我们一起来分析一下大名鼎鼎的 Leakcanary&#xff0c; 想必作为 Android 开发都多多少少接触过&#xff0c;新版本的 Leakcanary 也用 Kotlin 重写了一遍&#xff0c;最近详细查看了下源码&#xff0c;分享一下。 tips&#xff1a;本来是只想分析下内存泄漏检测部分&#xff…

c语言Have Fun with Numbers

题目 Have Fun with Numbers Notice that the number 123456789 is a 9-digit number consisting exactly the numbers from 1 to 9, with no duplication. Double it we will obtain 246913578, which happens to be another 9-digit number consisting exactly the numbers …

2016年iOS公开可利用漏洞总结

0x00 序 iOS的安全性远比大家的想象中脆弱&#xff0c;除了没有公开的漏洞以外&#xff0c;还有很多已经公开并且可被利用的漏洞&#xff0c;本报告总结了2016年比较严重的iOS漏洞&#xff08;可用于远程代码执行或越狱&#xff09;&#xff0c;希望能够对大家移动安全方面的工…

【LeetCode】139. 单词拆分

139. 单词拆分&#xff08;中等&#xff09; 思路 首先将大问题分解成小问题&#xff1a; 前 i 个字符的子串&#xff0c;能否分解成单词&#xff1b;剩余子串&#xff0c;是否为单个单词&#xff1b; 动态规划的四个步骤&#xff1a; 确定 dp 数组以及下标的含义 dp[i] 表示 s…

Zero-ETL、大模型和数据工程的未来

编者按&#xff1a;本文探讨了数据工程领域的未来趋势和挑战&#xff0c;以及其不断变化、甚至经常出现“重塑”的特点。在数据工程领域&#xff0c;大数据的性能、容量提升总是有一定的上限&#xff0c;每一次进步都会带来一定的技术提升&#xff0c;从而提高上限。但是很快我…

解锁采购系统数字升级?来看看云时通SRM!

疫情影响下&#xff0c;全球经济一体化使得企业在供应商的管理上面临着巨大挑战&#xff0c;传统采购方法已经不足以支持企业管理和竞争&#xff0c;企业采购管理急需数字化转型。 相较于传统采购方法&#xff0c;目前成功的供应商系统管理&#xff0c;还需要具有更多的市场要…

IT圈最近比较火热的技术都是哪些?

前言 如果现在要问什么行业最火&#xff0c;毋庸置疑&#xff0c;会有很多人第一反应回答是IT行业。众所周知&#xff0c;近些年互联网行业的高速发展&#xff0c;IT技术不断推陈出新&#xff0c;各种技术更新迭代周期越来越快&#xff0c;涌入IT技术开发的人员也是成倍增长&am…

QT基础铺垫

1.qt定位 qt在整个课程体系中起到以下作用 1.c理论的实践课 2.图形用户界面GUI开发 3.一个独立的就业方向 3.qt特性 qt经常被当作是一个基于c语言的gui开发框架&#xff0c;但是这并不是qt的全部&#xff0c;除了开发界面外&#xff0c;qt还包含了很多其他功能&#xff1…