排序算法(3)——归并排序、计数排序

目录

1. 归并排序 

1.1 递归实现 

1.2 非递归实现

1.3 归并排序特性总结

 2. 计数排序

代码实现

 3. 总结


1. 归并排序 

基本思想: 归并排序(merge sort)是建立在归并操作上的一种有效的排序算法,该算法是采用分治法(Divide and Conquer)的一个非常典型的应用。将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。若将两个有序表合并成一个有序表,称为二路归并。

划分阶段:通过递归不断地将数组从中间点分开。

合并阶段:当数组中只有一个元素时停止划分,开始合并,将左右两个较短的子序列按照一定的规则合并。 

拆分序列 

(left+right)/2为中间点,将数组拆分成两个无序序列,循环拆分,拆分到最后,每个序列都是单独的一个元素,再开始归并。

合并有序序列 

从两个数列的第一个数开始,谁小就先取谁,放到临时数组中,指针再向后走,接着进行比较,如果一个数列走完了,另一个数列还有剩余,直接把剩余的元素依次取下来放到临时数组中。

 递归展开图

通过观察,归并排序与二叉树后序遍历递归顺序相同。 

1.1 递归实现 

//归并
void merge(int* a, int* tmp, int left, int mid, int right)//左半区的起始位置  中间点  右半区结束位置
{int begin1 = left, end1 = mid;int begin2 = mid + 1, end2 = right;    //[left,mid]  [mid+1,right] //划分成[left,mid-1][mid,right]会造成死循环!int i = left;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 + left, tmp + left, (right - left + 1) * sizeof(int));
}
//划分和归并
void Msort(int* a, int* tmp, int left, int right)
{if (left >= right)return;//找中间点int mid = (left + right) / 2;	//[begin,mid-1]   [mid,end]	//递归划分左半区域Msort(a, tmp, left, mid);//递归划分右半区域Msort(a, tmp, mid + 1, right);//归并merge(a, tmp, left, mid, right);}
//归并排序入口
void MergeSort(int* a, int n)
{//开辟一个辅助数组int* tmp = (int*)malloc(sizeof(int) * n);if (tmp == NULL){perror("malloc fail!");return;}Msort(a, tmp, 0, n - 1);//划分和归并free(tmp);//释放空间tmp = NULL;
}

复杂度分析 

时间复杂度:O(n*logn)  数组划分的深度是 logn,而在每一层递归中,合并操作的时间复杂度是 O(n),所以,总的时间复杂度为O(n*logn),这个时间复杂度是在最好、最坏和平均情况下都成立的,因为归并排序不依赖于原始数组的顺序。

空间复杂度:O(n)   归并排序需要在归并过程中需要与原数组同样的存储空间存放归并结果,还需要额外的空间来存储递归调用的栈,占用空间 n+logn 

1.2 非递归实现

定义一个变量gap,规定gap为每组归并数据的数据个数,gap=1,2,4,8,16……1个和一个归并,2个和2个归并,4个和4个归并,依次循环下去直到gap>n。

//归并排序——非递归
void MergeSortNonRecur(int* a, int n)
{//开辟一个辅助数组int* tmp = (int*)malloc(sizeof(int) * n);if (tmp == NULL){perror("malloc fail!");return;}//规定gap为每组归并数据的数据个数int gap = 1;while (gap < n){for (int 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)//第二组begin2没越界,end2越界了,修正一下,继续归并{end2 = n - 1;}printf("[%d][%d][%d][%d]  ", begin1, end1, begin2, end2);int k = i;while (begin1 <= end1 && begin2 <= end2){if (a[begin1] <= a[begin2]){tmp[k++] = a[begin1++];}else{tmp[k++] = a[begin2++];}				}while (begin1 <= end1){tmp[k++] = a[begin1++];}while (begin2 <= end2){tmp[k++] = a[begin2++];}memcpy(a + i, tmp + i, sizeof(int) * (end2 - i + 1));}printf("\n");gap *= 2;}free(tmp);tmp = NULL;
}

上面两条 if 语句是为了处理一下越界情况: 

非递归的迭代方法,避免了递归时深度为log₂n的栈空间,空间只是用到归并临时用的tmp数组,空间复杂度为O(n),避免递归在时间性能上有一定的提升,使用归并排序时,尽量考虑用非递归方法。

1.3 归并排序特性总结

时间复杂度:O(n*logn)

空间复杂度:O(n)

稳定性:稳定

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

适用性:归并排序适用于各种数据规模的排序,而且对于大规模数据的排序效果较好。它的时间复杂度稳定在O(nlogn),不会因为数据规模的增大而导致时间复杂度的增加。由于其空间复杂度较高,通常在内排序中不会使用归并排序,而是选择快速排序。在外排序中,对于无法一次性加载到内存的大规模数据进行排序,归并排序则是一个很好的选择。

 2. 计数排序

计数排序是一种非比较型排序算法,适用于一定范围内的整数排序。在计数排序中,我们不直接比较元素的大小,而是利用数组索引来统计每个元素的出现次数。

思想:计数排序又称为鸽巢原理,是对哈希直接定址法的变形应用。

操作步骤:

1. 统计相同元素出现次数

2. 根据统计的结果将序列回收到原来的序列中

要根据range的范围来创建统计数组count,比如上面数组值的范围是99~109,如果按照最大值创建,那就需要长度109的数组,会造成极大的空间浪费。 所以变形一下,每个元素在统计数组的位置为减去最小值之后的下标位置。

calloc函数开辟空间的时候会把空间的每个字节都初始化为0,这就不需要我们手动去初始化了,没有出现的数字在统计数组中就是0。

代码实现

void CountSort(int* a, int n)
{int min = a[0], max = a[0]; //找最大和最小值for (int i = 1; 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!");return;}//统计次数for (int i = 0; i < n; i++){count[a[i] - min]++;}//排序int j = 0;for (int i = 0; i < range; i++){while (count[i]--){a[j++] = i + min;}}free(count);count = NULL;   
}

计数排序是稳定的排序算法,时间复杂度为 O(n + range),空间复杂度为 O(range)。但是,计数排序对数据的要求较为严格,只适合整数和范围集中的数据。

 3. 总结

排序方法时间复杂度(平均)时间复杂度(最好)时间复杂度(最坏)空间复杂度稳定性
冒泡排序O(n²)O(n)O(n²)O(1)稳定
简单选择排序O(n²)O(n²)O(n²)O(1)不稳定
直接插入排序O(n²)O(n)O(n²)O(1)稳定
希尔排序O(nlogn)~O(n²)O(n^1.3)O(n²)O(1)不稳定
堆排序O(nlogn)O(nlogn)O(nlogn)O(1)不稳定
归并排序O(nlogn)O(nlogn)O(nlogn)O(n)稳定
快速排序O(nlogn)O(nlogn)O(n²)O(logn)~O(n)不稳定

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

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

相关文章

GIN

gin是什么 Gin 是一个用 Go (Golang) 编写的 HTTP Web 框架。 它具有类似 Martini 的 API&#xff0c;但性能比 Martini 快 40 倍。如果你需要极好的性能&#xff0c;使用 Gin 吧。 特点&#xff1a;gin是golang的net/http库封装的web框架&#xff0c;api友好&#xff0c;注…

基于asp.net游乐园管理系统设计与实现

博主介绍&#xff1a;专注于Java&#xff08;springboot ssm 等开发框架&#xff09; vue .net php python(flask Django) 小程序 等诸多技术领域和毕业项目实战、企业信息化系统建设&#xff0c;从业十五余年开发设计教学工作 ☆☆☆ 精彩专栏推荐订阅☆☆☆☆☆不然下次找…

PH热榜 | 2024-12-16

1. Animate AI 标语&#xff1a;动画系列剧集AI视频生成器 介绍&#xff1a;Animate AI 是一体化AI动画视频生成器&#xff0c;可以快速、轻松制作出动画系列视频。今天就免费开启你的创意之旅吧&#xff01;轻松搞定&#xff01; 产品网站&#xff1a; 立即访问 Product H…

VScode使用教程(菜鸟版)

目录 1.VScode是什么&#xff1f; 2.VScode的下载和安装&#xff1f; 2.1下载和安装 下载路径&#xff1a; 安装流程&#xff1a; 一、点击【Download for Windows】 二、等一小会儿的下载&#xff0c;找到并双击你下载好的.exe文件&#xff0c;开始进入安装进程 三、点…

【前端基础篇】JavaScript之DOM介绍

文章目录 前言WebAPI背景知识什么是WebAPI什么是APIAPI参考文档 DOM基本概念什么是DOMDOM树查找HTML元素方法概览1. document.getElementById(id)2.document.getElementsByTagName(name)3. document.getElementsByClassName(name)4. document.querySelector(CSS选择器)5. docum…

LabVIEW起落架震台检测

在现代飞机制造与维护过程中&#xff0c;起落架的性能测试是保障飞机安全的重要环节。通过LabVIEW开发的起落架小落震台检测系统&#xff0c;通过模拟飞机着陆过程&#xff0c;准确捕捉起落架在着陆时承受的各种动力学特性和应力响应&#xff0c;有效提升起落架设计的精度与可靠…

(长期更新)《零基础入门 ArcGIS(ArcMap) 》实验一(下)----空间数据的编辑与处理(超超超详细!!!)

续上篇博客&#xff08;长期更新&#xff09;《零基础入门 ArcGIS(ArcMap) 》实验一&#xff08;上&#xff09;----空间数据的编辑与处理&#xff08;超超超详细&#xff01;&#xff01;&#xff01;&#xff09;-CSDN博客 继续更新 目录 什么是拓扑&#xff1f; 1.3.5道路拓…

libaom 源码分析:熵编码模块介绍

AV1 熵编码原理介绍 关于AV1 熵编码原理介绍可以参考:AV1 编码标准熵编码技术概述libaom 熵编码相关源码介绍 函数流程图 核心函数介绍 av1_pack_bitstream 函数:该函数负责将编码后的数据打包成符合 AV1 标准的比特流格式;包括写入序列头 OBU 的函数 av1_write_obu_header…

pset4filter less: helpers.c

&#xff08;&#xff14;&#xff09;blur function 简单画图熟悉一下要做什么 可以看到3种情况&#xff0c;顶格&#xff0c;边界&#xff0c;里面如果分开算的话&#xff0c;是真的麻烦&#xff1b;但是当时还真的没有想到更好的&#xff0c;就先写一写&#xff08;此处摘取…

【java】规则引擎

Java 规则引擎&#xff1a;Easy Rules 简介 优点 通过 Easy Rules 引入规则引擎的主要优点是解耦业务规则和代码逻辑&#xff1a; 无需修改代码&#xff0c;只需更新数据库中的规则即可。提供灵活、可扩展和可维护的方式处理复杂业务逻辑。 规则定义 Easy Rules 支持多种形…

满足你的个性化需求!ChatGLM4 模型微调教程

一、 大模型 ChatGLM4 简介 GLM-4-9B 是智谱 AI 推出的最新一代预训练模型 GLM-4 系列中的开源版本。 在语义、数学、推理、代码和知识等多方面的数据集测评中&#xff0c; GLM-4-9B 及其人类偏好对齐的版本 GLM-4-9B-Chat 均表现出超越 Llama-3-8B 的卓越性能。 除了能进行多…

TCP基础了解

什么是 TCP &#xff1f; TCP 是面向连接的、可靠的、基于字节流的传输层通信协议。 面向连接&#xff1a;一定是「一对一」才能连接&#xff0c;不能像 UDP 协议可以一个主机同时向多个主机发送消息&#xff0c;也就是一对多是无法做到的&#xff1b; 可靠的&#xff1a;无论…

如何测量分辨率

一、什么是分辨率&#xff1f; 分辨率指的是分清物体细节的能力。分辨率是一个成像系统还原空间频率的能力。一些人只是简单的用分辨率去描述极限分辨率&#xff0c;但是相机在在不同的对比度的情况下还原低&#xff0c;中和高频率的能力&#xff0c;也可以显示全面综合的信息。…

HCIA-Access V2.5_4_1_1路由协议基础_IP路由表

大型网络的拓扑结构一般会比较复杂&#xff0c;不同的部门&#xff0c;或者总部和分支可能处在不同的网络中&#xff0c;此时就需要使用路由器来连接不同的网络&#xff0c;实现网络之间的数据转发。 本章将介绍路由协议的基础知识、路由表的分类、静态路由基础与配置、VLAN间…

气象与旅游之间的关系,如果借助高精度预测提高旅游的质量

气象与旅游之间存在密切的关系,天气条件直接影响旅游者的出行决策、旅游体验和安全保障。通过高精度气象预测技术,可以有效提升旅游质量,为游客和旅游行业带来显著的优势。 1. 提高游客出行决策效率 个性化天气服务:基于高精度气象预测,旅游平台可以提供个性化的天气预报服…

【计算机视觉基础CV】03-深度学习图像分类实战:鲜花数据集加载与预处理详解

本文将深入介绍鲜花分类数据集的加载与处理方式&#xff0c;同时详细解释代码的每一步骤并给出更丰富的实践建议和拓展思路。以实用为导向&#xff0c;为读者提供从数据组织、预处理、加载到可视化展示的完整过程&#xff0c;并为后续模型训练打下基础。 前言 在计算机视觉的深…

数据结构之线性表1

2.1 线性表的定义和基本操作 1.线性结构的特点是&#xff1a;在数据元素的非空有限集中&#xff0c; &#xff08;1&#xff09;存在惟一的一个被称做“第一个”的数据元素&#xff1b; &#xff08;2&#xff09; 存在惟一的一个被称做“最后一个”的数据元素&#xff1b; &a…

信息安全实训室网络攻防靶场实战核心平台解决方案

一、引言 网络安全靶场&#xff0c;作为一种融合了虚拟与现实环境的综合性平台&#xff0c;专为基础设施、应用程序及物理系统等目标设计&#xff0c;旨在向系统用户提供全方位的安全服务&#xff0c;涵盖教学、研究、训练及测试等多个维度。随着网络空间对抗态势的日益复杂化…

关于分页的样式问题

在最近写网页的时候遇到了一个关于样式的问题&#xff0c;今天我来跟大家来说一下。像是分页中的颜色效果&#xff0c;斑马纹颜色要注意颜色不要过于深。 这种的颜色就有一点深看着很不舒服&#xff0c;应将当前的颜色改为淡一点的&#xff0c;也可以利用rgba调整透明度&#x…

一分钟快速了解什么是AEO海关认证

一分钟快速了解什么是AEO海关认证——这一术语&#xff0c;对于国际贸易领域的从业者而言&#xff0c;无疑是一个充满分量与价值的标签。AEO&#xff0c;即“Authorized Economic Operator”&#xff0c;中文译为“经认证的经营者”&#xff0c;是海关对信用状况、守法程度和安…