排序-归并排序与计数排序

文章目录

    • 一、归并排序
      • 1、概念
      • 2、过程
      • 3、代码实现
      • 4、复杂度
      • 5、稳定性
    • 二、 计数排序
      • 1、思路
      • 2、代码实现
      • 3、复杂度:
      • 4、稳定性


一、归并排序

1、概念

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

在这里插入图片描述

2、过程

假设前提:
左半区间 ->有序
右半区间 ->有序
怎么使左右排序呢?
当只剩下一个元素时我们可以默认其为有序的,所以我们可以利用递归将数组中的元素划分为一个,再两组两组合并,以此类推。
归并,依次对比取小的放到新到临时数组中,完成排序后再将临时数组的数据拷贝回原来的数组
过程图:
在这里插入图片描述

3、代码实现

递归:

void Print(int* arr, int n) {

	for (int i = 0; i < n; i++)
		printf("%d ", arr[i]);
}
//递归


void _MergeSort(int *a,int *t,int left,int right) {
	//结束条件
	if (left >= right)
		return;

	int mid = (left + right) >> 1;//取中间数,划分区间
	//[left  mid]  [mid+1  right]
	//递归
	_MergeSort(a, t, left, mid);
	_MergeSort(a, t, mid + 1, right);
	//回归
	int begin1 = left, end1 = mid;//左区间
	int begin2 = mid + 1 , end2  = right;//右区间
	//临时数组下标->对应的是数组a的下标
	int index = left;
	//当左区间或者右区间,遍历完了就结束了
	while (begin1 <= end1 && begin2 <= end2) {
	//选择小的放进临时数组
		if (a[begin1] < a[begin2])
			t[index++] = a[begin1++];
		else
			t[index++] = a[begin2++];
	}
	//判断左右两边是否都空了,不为空将后面补上
	while (begin1 <= end1)
		t[index++] = a[begin1++];
	while (begin2 <= end2)
		t[index++] = a[begin2++];
		//最后拷贝回去
	for (int i = left; i <= right; ++i)
		a[i] = t[i];
 }
void MergeSort(int* a, int n) {
	int* t = (int*)malloc(sizeof(int) * n);
	_MergeSort(a, t, 0, n - 1);
	
	free(t);
}

int main() {
	int a[] = { 3710962385 };
	MergeSort(a, sizeof(a) / sizeof(a[0]));
	Print(a, sizeof(a) / sizeof(a[0]));
	return 0;
}

递归图(左边,先递后归):
在这里插入图片描述

非递归:
我们通过循环来实现非递归
(1)设置一个gap来划分归并个数,先设置gap=1,这样控制第一次是两个数合并,gap再乘2,来递增,当 gap>n(数组大小)时结束
(2)在合并的过程中可能出现两种情况
a.合并过程中右边没元素
如:
在这里插入图片描述
解决办法:因为已经排好了,直接打破循环即可
b,右边有元素但是不够
如:
在这里插入图片描述
解决办法:进行纠正,将右端的下标改为 n-1(数组大小-1)

代码实现:

//非递归

void MergeSortNonR(int* a, int* t,int n) {
	int gap= 1;//划分一次归并多少个元素
	//结束条件
	while (gap<n) {
		for (int i = 0; i < n; i += 2*gap) {
		//通过gap划分区间
			int begin1 = i, end1 = i + gap - 1;
			int begin2 = i + gap, end2 = i + gap * 2 - 1;
			//情况a,此时直接打破即可
			if (begin2 >= n)
				break;
				//情况b,进行纠正
			if (end2 >= n)
				end2 = n - 1;
 
			int index = i;//从控制的区间最小的位置开始
			//下面过程与递归过程一样
			while (begin1 <= end1 && begin2 <= end2) {
				if (a[begin1] < a[begin2])
					t[index++] = a[begin1++];
				else
					t[index++] = a[begin2++];
			}
			while (begin1 <= end1)
				t[index++] = a[begin1++];
			while (begin2 <= end2)
				t[index++] = a[begin2++];
			for (int j = i; j <= end2; j++)
				a[j] = t[j];
		}
		gap *= 2;//每次加倍
	}
}



void MergeSort(int* a, int n) {
	int* t = (int*)malloc(sizeof(int) * n);
	MergeSortNonR(a, t, n);
	free(t);
}

int main() {
	int a[] = { 6,3,7,1,9,5,2,8,0,4 };
	MergeSort(a, sizeof(a) / sizeof(a[0]));
	Print(a, sizeof(a) / sizeof(a[0]));
	return 0;
}

4、复杂度

时间复杂度:
(1)循环部分:N
(2)递归部分:因为每次都减半所以就是logN(以2为底)
所以时间复杂度为:O(N*logN)
空间复杂度:
因为要重新开辟一个数组,所以空间复杂度为O(N)

5、稳定性

在归并过程中相同的元素的顺序不会发生改变,所以是稳定的

二、 计数排序

1、思路

通过映射统计每个数出现的次数,再使用次数排序
如:
在这里插入图片描述
上述是以最大数去创建空间
但是如果遇到一个很大的数的话就需要我们创建空间时就会很浪费
如:
在这里插入图片描述
解决:找到范围,使用范围+1去创建临时空间

2、代码实现

//计数排序
void  CountSort(int* a, int n) {
	int max = a[0];
	int min = a[0];
	//求出数组的范围
	for (int i = 0; i < n; i++) {
		if (max < a[i])
			max = a[i];
		if (min > a[i])
			min = a[i];
	}
	int  t = max - min+1;
	//临时空间
	int* p = (int*)calloc(t,sizeof(int));
	//统计个数
	for (int j = 0; j < n; j++) {
		//a[j]-min当下标,我们下次直接加回min即可
		p[a[j] - min]++;
	}
	int i = 0;
	//按顺序拷贝回原来的数组
	for (int j = 0; j < t; j++) {
		
		while (p[j]) {
			a[i] = j + min;
			i++;
			p[j]--;
		}
	}
	free(p);
	p = NULL;
}

3、复杂度:

空间复杂度:因为要创建临时的空间,所以复杂度为O(N);
时间复杂度:O(N+t)

4、稳定性

在统计和重新排序过程中相同元素可能位置发生交换,所以为不稳定

以上就是我的分享了,如果有什么错误,欢迎在评论区留言。
最后,谢谢大家的观看!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Shell三剑客:正则表达式(元字符)

一、定义&#xff1a;元字符字符是这样一类字符&#xff0c;它们表达的是不同字面本身的含义 二、分类&#xff1a; 1、基本正则表达式元字符 # ^ 行首定位 [rootlocalhost ~]# grep root /etc/passwd root:x:0:0:root:/root:/bin/bash operator:x:11:0:operator:/root:/…

分布式块存储 ZBS 的自主研发之旅|元数据管理

重点内容 元数据管理十分重要&#xff0c;犹如整个存储系统的“大黄页”&#xff0c;如果元数据操作出现性能瓶颈&#xff0c;将严重影响存储系统的整体性能。如何提升元数据处理速度与高可用是元数据管理的挑战之一。SmartX 分布式存储 ZBS 采用 Log Replication 的机制&…

The Grid – Responsive WordPress Grid响应式网格插件

点击阅读The Grid – Responsive WordPress Grid响应式网格插件原文 The Grid – Responsive WordPress Grid响应式网格插件是一个高级 wordpress 网格插件&#xff0c;它允许您在完全可定制且响应迅速的网格系统中展示任何自定义帖子类型。 Grid WordPress 非常适合展示您的博…

信号与线性系统预备训练3——MATLAB软件在信号与系统中的应用初步

信号与线性系统预备训练3——MATLAB软件在信号与系统中的应用初步 The Preparatory training3 of Signals and Linear Systems 对应教材&#xff1a;《信号与线性系统分析&#xff08;第五版&#xff09;》高等教育出版社&#xff0c;吴大正著 一、目的 1.熟悉和回顾MATLAB…

Android动画

关于作者&#xff1a;CSDN内容合伙人、技术专家&#xff0c; 从零开始做日活千万级APP。 专注于分享各领域原创系列文章 &#xff0c;擅长java后端、移动开发、商业变现、人工智能等&#xff0c;希望大家多多支持。 目录 一、导读二、概览三、动画实现3.1 帧动画资源文件中实现…

会声会影2024中文旗舰版强悍来袭及视频片头怎么制作

​ 随着网络视频的蓬勃发展&#xff0c;越来越多的人开始涉足视频剪辑领域&#xff0c;毕竟技多不压身嘛。在众多剪辑软件中&#xff0c;剪映和会声会影是备受新手青睐的两种。那么&#xff0c;会声会影2024出来了吗&#xff1f; 会声会影2024中文旗舰版是一款加拿大公司Corel发…

cpp_03_引用_类型转换_温和强转_面向对象

1 引用的应用 1.1 数据传递--值传递 C语言只要涉及数据传递&#xff08;例如&#xff1a;初始化、赋值、传参、返回值&#xff09;&#xff0c; 都为值传递&#xff08;将数据复制一份给到目标内存&#xff09;。 // value.cpp 值传递&#xff1a;将数据复制一份给别人 #in…

C++软件调试与异常排查技术从入门到精通学习路线分享

目录 1、概述 2、全面了解引发C软件异常的常见原因 3、熟练掌握排查C软件异常的常见手段与方法 3.1、IDE调试 3.2、添加打印日志 3.3、分块注释代码 3.4、数据断点 3.5、历史版本比对法 3.6、Windbg静态分析与动态调试 3.7、使用IDA查看汇编代码 3.8、使用常用工具分…

正大杯获奖作品在哪可以看见

通过网盘分享的文件&#xff1a;2023年第十三届正大杯最新国家一等奖完整获奖作品报告等全套资料 链接:https://pan.baidu.com/s/1SPA4LumSCI4BZdCRXXnW6Q?pwdc8bj 提取码:c8bj 2023年第十三届最新正大杯国家一等奖完整获奖作品等全套资料获取方式链接https://ex59573j43x.fe…

造型精致的冰精灵充电头,充电效率高安全可靠,居家出行皆可用

随着大家对手机的依赖度越来越高&#xff0c;快速充电已经成为必不可少的需求。快充当然少不了支持快充的充电器&#xff0c;现在市面上的快充头很多&#xff0c;安全性和便携性是我们选择时的重点关注方向&#xff0c;我目前用的是战飞ZEFi冰精灵&#xff0c;这款产品有着独特…

〖大前端 - 基础入门三大核心之JS篇(56)〗- 内置构造函数

说明&#xff1a;该文属于 大前端全栈架构白宝书专栏&#xff0c;目前阶段免费&#xff0c;如需要项目实战或者是体系化资源&#xff0c;文末名片加V&#xff01;作者&#xff1a;哈哥撩编程&#xff0c;十余年工作经验, 从事过全栈研发、产品经理等工作&#xff0c;目前在公司…

Python 全栈体系【四阶】(六)

第四章 机器学习 五、线性模型 1. 概述 线性模型是自然界最简单的模型之一&#xff0c;它描述了一个&#xff08;或多个&#xff09;自变量对另一个因变量的影响是呈简单的比例、线性关系。例如&#xff1a; 住房每平米单价为 1 万元&#xff0c;100 平米住房价格为 100 万…

Python学习路线 - Python语言基础入门 - 函数进阶

Python学习路线 - Python语言基础入门 - 函数进阶 函数的多返回值多个返回值 函数的多种参数使用形式函数参数种类位置参数关键字参数缺省参数不定长参数位置传递 - 不定长关键字传递 - 不定长 函数作为参数传递lambda匿名函数 函数的多返回值 问: 如果一个函数如些两个return…

java.net.SocketException: Connection reset

背景 在我用socket进行TCP通信的时候&#xff0c;当我关闭client端时在服务端出现了Connection reset的异常。 一、问题 下面是异常信息&#xff1a; Exception in thread "Thread-12" java.lang.RuntimeException: java.net.SocketException: Connection reseta…

基于ssm居家养老系统论文

居家养老系统 摘要 随着信息技术在管理上越来越深入而广泛的应用&#xff0c;管理信息系统的实施在技术上已逐步成熟。本文介绍了居家养老系统的开发全过程。通过分析高校学生综合素质评价管理方面的不足&#xff0c;创建了一个计算机管理居家养老系统的方案。文章介绍了居家…

数据手册Datasheet解读-MOS管笔记

数据手册Datasheet解读-MOS管笔记 NMOS应用场景一般特征第一个参数Vdss第二、三个参数Rds&#xff08;on&#xff09;、IdMOS管的散热绝对最大额定值第一个参数-Vd第二个参数-Vdgr第三个参数-Vg(栅源电压)第四个参数-Id第五个参数-Idm第六个参数-Ptot第七个参数-Viso第七和八的…

tomcat错误

Error running Tomcat8: Address localhost:1099 is already in use window环境&#xff0c;打开cmd netstat -ano | findstr :1099发现对应PID为24732 结束PID taskkill /PID 24732 /F

2023年度佳作:AIGC、AGI、GhatGPT、人工智能大语言模型的崛起与挑战

目录 前言 01 《ChatGPT 驱动软件开发》 内容简介 02 《ChatGPT原理与实战》 内容简介 03 《神经网络与深度学习》 04 《AIGC重塑教育》 内容简介 05 《通用人工智能》 目  录 前言 2023年是人工智能大语言模型大爆发的一年&#xff0c;一些概念和英文缩写也在这一…

字符迷宫(期末考模拟题)

很有趣的一道题 难点主要在于对于 * 的处理 题目描述的是可以多次匹配相同的字母&#xff0c;这就涉及到两个方面&#xff1a; 一是这个匹配的相同的字母如何储存 二是当你’ * ‘位置递归结束的时候&#xff0c;你该什么时候变回‘ * ’号 这里给出我的思路&#xff0c;如…

Zookeeper教程

Zookeeper教程 1、Zookeeper CLI ZooKeeper命令行界面CLI用于与ZooKeeper集合进行交互以进行开发。它有助于调试和解决不同的选项。 要执行ZooKeeper CLI操作&#xff0c;首先打开ZooKeeper服务器bin/zkServer.sh start&#xff0c;然后打开ZooKeeper客户端 bin/zkCli.sh。…