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

news2025/1/10 3:17:28

文章目录

    • 一、归并排序
      • 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、稳定性

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

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

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

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

相关文章

车载导航系统UI界面,可视化大屏设计(PS源文件)

大屏组件可以让UI设计师的工作更加便捷&#xff0c;使其更高效快速的完成设计任务。现分享车载导航系统科技风蓝黑简约UI界面、车载系统UI主界面、车载系统科技风UI界面、首页车载系统科技感界面界面的大屏Photoshop源文件&#xff0c;开箱即用&#xff01; 若需 更多行业 相关…

数据库动态视图和存储过程报表数据管理功能设计

需求&#xff1a;需要将ERP的报表数据挪到OA中&#xff0c;但是OA表单设计不支持存储过程动态传参&#xff0c;所以需要设计一个系统&#xff0c;可以手动配置&#xff0c;动态显示原本ERP的报表数据&#xff0c;ERP报表是存在数据库的视图和存储过程中 思路&#xff1a;因为E…

算法复习——6种排序方法的简单回顾

算法复习——6种排序方法的简单回顾 常见排序方法&#xff1a;冒泡排序、选择排序、插入排序、堆排序、归并排序、快速排序的简单回顾 冒泡排序 重复“从序列右边开始比较相邻两个数字的大小,再根据结果交换两个数字的位置” 在冒泡排序中&#xff0c;第 1 轮需要比较 n - 1…

整理b站黑马程序员C++课程中对于计算机视觉学习有所帮助的知识点。(重点用*标出)

文章目录 1、注释2、变量3、常量4、标识符5、整型 浮点型 字符型 字符串 布尔6、输入 输出7、逻辑运算法8、 程序流程结构9、三目运算符10、switch语句11、循环语句12、跳转语句13、*数组13.1一维数组名 14、二维数组15、**函数15.1、函数的调用15.2、函数的声明15.3、函数份文…

Android camera的metadata

一、实现 先看一下metadata内部是什么样子&#xff1a; 可以看出&#xff0c;metadata 内部是一块连续的内存空间。 其内存分布大致可概括为&#xff1a; 区域一 &#xff1a;存 camera_metadata_t 结构体定义&#xff0c;占用内存 96 Byte 区域二 &#xff1a;保留区&#x…

HarmonyOS--基础组件TextInput

TextInput 官方文档 TextInput组件https://developer.harmonyos.com/cn/docs/documentation/doc-references-V3/ts-basic-components-textinput-0000001427584864-V3#ZH-CN_TOPIC_0000001523968610__%E5%AD%90%E7%BB%84%E4%BB%B6 文本输入框组件 接口 TextInput(value?:…

【Python】用Python发邮件

准备工作 以新浪邮箱为例&#xff0c;进入账号管理&#xff0c;打开授权码并保存下来 用到的包 import smtplib from email.header import Header from email.mime.text import MIMEText 账号授权码准备 这里用的是前面记录的授权码&#xff0c;不是登录密码哦 email_hostsm…

40G AOC线缆全系列产品知识详解

40G AOC&#xff08;Active Optical Cable&#xff09;线缆作为高速数据传输的重要组成部分&#xff0c;在现代通信和数据中心应用中扮演着重要角色。本期文章我们将从其基本原理、应用领域、优势特点等方面对ETU-LINK 40G AOC全系列产品进行解析。 一、40G AOC全系列产品解析…

Facebook广告投放常见错误

在进行Facebook广告投放时&#xff0c;很容易犯一些常见的错误。这些错误可能导致广告投资的浪费&#xff0c;影响广告效果并降低回报。本文小编讲一些常见的Facebook广告投放错误&#xff0c;以及如何避免它们。 1、不明确目标受众 广告的成功与否很大程度上取决于你选择的目…

基于Java+Swing+mysql学生选课成绩信息管理系统

基于JavaSwingmysql学生选课成绩信息管理系统 一、系统介绍二、功能展示三、项目相关3.1 乱码问题3.2 如何将GBK编码系统修改为UTF-8编码的系统&#xff1f; 四、其它1.其他系统实现 五、源码下载 一、系统介绍 学生教师信息管理、年级班级信息管理、课程信息管理、选课、成绩…

怎么制作一个微信小程序商城

随着移动互联网的普及&#xff0c;越来越多的商家开始关注线上销售。微信小程序商城作为一种便捷、实用的线上销售平台&#xff0c;受到了广大商家的青睐。本文将详细介绍如何制作一个微信小程序商城。 一、登录乔拓云平台进入后台 首先&#xff0c;我们需要登录乔拓云平台&am…

亚信科技AntDB数据库——深入了解AntDB-M元数据锁的相关概念

AntDB-M在架构上分为两层&#xff0c;服务层和存储引擎层。元数据的并发管理集中在服务层&#xff0c;数据的存储访问在存储引擎层。为了保证DDL操作与DML操作之间的一致性&#xff0c;引入了元数据锁&#xff08;MDL&#xff09;。 AntDB-M提供了丰富的元数据锁功能&#xff…

服务器感染了.DevicData-D-XXXXXXXX勒索病毒,如何确保数据文件完整恢复?

引言&#xff1a; 勒索病毒成为网络安全的严峻挑战&#xff0c;而最新的.DevicData-D-XXXXXXXX勒索病毒更是引起广泛关注。本文将深入介绍.DevicData-D-XXXXXXXX勒索病毒的特征&#xff0c;提供恢复被其加密的数据文件的方法&#xff0c;并分享预防措施&#xff0c;以确保您的数…

【算法题】打印任务排序(js)

输入: 1,2,2 输出&#xff1a;2,0,1 说明:队列头部任务的优先级为1&#xff0c;被移到队列尾部&#xff1b;接着顺序打印两个优先级为2的任务&#xff0c;故其序号分别为0和1&#xff1b;最后打印剩下的优先级为1的任务&#xff0c;其序号为2 解法&#xff1a; const str &q…

_pickle.PicklingError: Can‘t pickle : import of module failed

有问题 没问题的 python - pickle cant import a module that exists? - Stack Overflow

1311:【例2.5】求逆序对 归并排序

1311&#xff1a;【例2.5】求逆序对 【题目描述】 给定一个序列a1,a2,…,an&#xff0c;如果存在i<j并且ai>aj&#xff0c;那么我们称之为逆序对&#xff0c;求逆序对的数目。 【输入】 第一行为n,表示序列长度&#xff0c;接下来的n行&#xff0c;第i1行表示序列中的第…

idea中定时+多数据源配置

因项目要求,需要定时从达梦数据库中取数据,并插入或更新到ORACLE数据库中 1.pom.xml <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0" xmlns:xsi"http://www.w3.org/2001/XMLSchema-…

01读《物联网安全研究综述:威胁、检测与防御》随笔

01读《物联网安全研究综述&#xff1a;威胁、检测与防御》随笔 摘要3 研究现状3.1 安全威胁3.1.1 云平台访问控制缺陷3.1.2 云平台恶意应用3.1.3 云平台实体和应用交互漏洞3.1.4 通信协议漏洞3.1.5 通信流量侧信道信息泄露3.1.6 设备固件漏洞3.1.7 基于语音信道的攻击3.1.8 基于…

SpringBoot核心功能-temp

yml&类配置 Configuration-processor

软文写作需要避免的四大误区,媒介盒子分享

有不少企业在做软文推广时在文案上容易踩坑&#xff0c;导致推广不起效。今天媒介盒子就来和大家聊聊软文写作中需要避免的四大类型&#xff0c;帮助企业提高软文推广效率。 一、 文案没有核心点 一篇软文的价值在于软文阐述的核心点&#xff0c;若没有一个核心点加以细化撰写…