时间复杂度接近O(n)的三种排序算法

news2024/11/15 11:59:59

1.桶排序

桶排序,顾名思义,会用到“桶”,核心思想是将要排序的数据分到几个有
序的桶里,每个桶内的数据再单独进行排序。桶内排完序之后,再把每个桶内的数据按照顺序依次
取出,组成的序列就是有序的了。

桶排序对要排序数据的要求是非常苛刻的。
首先,要排序的数据需要很容易就能划分成m个桶,并且,桶与桶之间有着天然的大小顺序。这样
每个桶内的数据都排序完之后,桶与桶之间的数据不需要再进行排序。

其次,数据在各个桶之间的分布是比较均匀的。如果数据经过桶的划分之后,有些桶内的数据非常
多,有些非常少,很不平均,那桶内数据排序的时间复杂度就不是常量级了。在极端情况下,如果
数据都被划分到一个桶内,那就退化为O(nlogn)的排序算法了。

桶排序比较适合用在外部排序中。所谓的外部排序就是数据存储在外部磁盘中,数据量比较大,内
存有限,无法将数据全部加载到内存中。
在这里插入图片描述

/**
 * 桶排序
 * 原地排序:否
 * 稳定排序:是
 * 空间复杂度:
 * 时间复杂度:O(n)
 */
public class BucketSort {
	public static void main(String[] args) {
		int[] arr = { 1, 45, 32, 23, 22, 31, 47, 24, 4, 15 };
		bucketSort(arr);
		System.out.println(Arrays.toString(arr));
	}
	
	//存数区间0-9,10-19,20-29,30-39,40-49
	public static void bucketSort(int[] arr) {
		ArrayList bucket[] = new ArrayList[5];
		
		//初始化桶
		for(int i=0;i<bucket.length;i++) {
			bucket[i] = new ArrayList<Integer>();
		}
		
		//像桶内放入数据
		for(int i=0;i<arr.length;i++) {
			int index = arr[i]/10;
			bucket[index].add(arr[i]);
		}
		
		int index = 0;
		for(int i=0;i<bucket.length;i++) {
			bucket[i].sort(null);//对每个桶进行排序
			for(int j=0;j<bucket[i].size();j++) {
				arr[index++] = (int) bucket[i].get(j);
			}
		}
	}
}

2.计数排序

计数排序可以理解为是桶排序的一种特殊情况。当要排序的n个数据,所处的范围并不大的
时候,比如最大值是k,我们就可以把数据划分成k个桶。每个桶内的数据值都是相同的,省掉了桶
内排序的时间。

计数排序只能用在数据范围不大的场景中,如果数据范围k比要排序的数据n大很多,
就不适合用计数排序了。而且,计数排序只能给非负整数排序,如果要排序的数据是其他类型的,
要将其在不改变相对大小的情况下,转化为非负整数。
假定有原始数组A[8],它们分别是:2,5,3,0,2,3,0,3。数据范围从0-5

先用一个数组大小为6的数组C来存储在k值上数据有几个。
在这里插入图片描述

接着对数组顺序求和,C数组存储的数据就变成了,C[k]里存储小于等于分数k的的数据。
在这里插入图片描述

定义临时数组R,依次扫描数组原始数组A,将数据A入到R[C[k]]位置上,同时C[k]位置上的数据要减掉1,最后将R数组复制到原始数组A中。
在这里插入图片描述

/**
 * 计数排序
 * 原地排序:否
 * 稳定排序:是
 * 空间复杂度:O(k+n) k为数据范围大小
 * 时间复杂度:O(n+k)
 */
public class CountSort {
	public static void main(String[] args) {
		int[] arr = new int[]{5,3,1,3,2,8,6,9,10,4,6,4,8,10,7,4,2,1,6,7};
		CountingSort(arr,arr.length);
		System.out.println(Arrays.toString(arr));
	}
	
	public static void CountingSort(int[] a,int n) {
		if(n<=-1) return;
		
		//查找数组中最大值
		int max = a[0];
		for(int i=1;i<a.length;i++) {
			if(max<a[i]) {
				max = a[i];
			}
		}
		
		//申请一个计数数组C下标为0到max
		int[] c = new int[max+1];
		for(int i=0;i<c.length;i++) {
			c[i] = 0;
		}
		
		//计算每个元素的个数,放入数组C中
		for(int i=0;i<a.length;i++) {
			c[a[i]]++;
		}
		
		//依次累加
		for(int i=0;i<max;i++) {
			c[i+1] = c[i]+c[i+1];
		}
		
		//临时数组R,存储排序之后的数组
		int[] r = new int[a.length];
		//计数排序的关键步骤
		for(int i=a.length-1;i>=0;i--) {
			int index = c[a[i]]-1;
			r[index] = a[i];
			c[a[i]]--;
		}
		
		//将结果拷贝给a数组
		for(int i=0;i<a.length;i++) {
			a[i] = r[i];
		}
	}
}

3.基数排序

先按照数据最后以位来排序,然后,再按照倒数第二位重新排序,以此类推,最后按照第一位重新排序。经过多次排序之后,数据就都有序了。如果数据长度不一致,需要补齐数据到相同长短。

基数排序对要排序的数据是有要求的,需要可以分割出独立的“位”来比较,而且位之间有递进的关系,如果a数据的高位比b数据大,那剩下的低位就不需要较了。除此之外,每一位的数据范围不能太大,才可以用线性排序算法来排序,否则,基数排序的时间复杂度就无法做到O(n)了。
在这里插入图片描述

/**
 * 基数排序
 * 原地排序:否
 * 稳定排序:是
 * 空间复杂度:O(d+n) k为数据范围大小
 * 时间复杂度:O(dn) d是维数
 */
public class RadixSort {
	public static void main(String[] args) {
		int[] arrs = new int[] {153,26,78,342,123,241,96};
		
		int max = getMaxData(arrs);
		
		for(int exp=1;max/exp>0;exp=exp*10) {
			CountingSort(arrs,arrs.length,exp);
			System.out.println(Arrays.toString(arrs));
		}
	}
	
	public static int getMaxData(int[] a) {
		//查找数组中最大值
		int max = a[0];
		for(int i=1;i<a.length;i++) {
			if(max<a[i]) {
				max = a[i];
			}
		}
		return max;
	}
	
	public static void CountingSort(int[] a,int n,int exp) {
		if(n<=-1) return;
		
		//查找数组中最大值
		int max = (a[0]/exp)%10;
		for(int i=1;i<a.length;i++) {
			if(max<(a[i]/exp)%10) {
				max = (a[i]/exp)%10;
			}
		}
		
		//申请一个计数数组C下标为0到max
		int[] c = new int[max+1];
		for(int i=0;i<c.length;i++) {
			c[i] = 0;
		}
		
		//计算每个元素的个数,放入数组C中
		for(int i=0;i<a.length;i++) {
			c[(a[i]/exp)%10]++;
		}
		
		//依次累加
		for(int i=0;i<max;i++) {
			c[i+1] = c[i]+c[i+1];
		}
		
		//临时数组R,存储排序之后的数组
		int[] r = new int[a.length];
		//计数排序的关键步骤
		for(int i=a.length-1;i>=0;i--) {
			int index = c[(a[i]/exp)%10]-1;
			r[index] = a[i];
			c[(a[i]/exp)%10]--;
		}
		
		//将结果拷贝给a数组
		for(int i=0;i<a.length;i++) {
			a[i] = r[i];
		}
	}

}

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

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

相关文章

UnrealEngine - 网络同步入门

1 网络同步机制 UE 提供了强大的网络同步机制&#xff1a; RPC &#xff1a;可以在本地调用&#xff0c;对端执行属性同步&#xff1a;标记一个属性为 UPROPERTY(Replicated) 就可以自动将其修改后的值同步到客户端移动复制&#xff1a;Actor 开启了移动复制后会自动复制位置…

区块链实验室(14) - 编译FISCO-BCOS

FISCO-BCOS是一种区块链平台&#xff0c;与Hyperledger和Ethereum有些不同&#xff0c;详见FISCO BCOS 区块链 编译FISCO BCOS源码的目的是修改或者新增其中功能模块&#xff0c;进行对比实验&#xff0c;验证新想法、新创意的效果。编译的步骤很简单&#xff0c;按技术文档一…

postman----传参格式(json格式、表单格式)

本文主要讲解postman使用post请求方法的2中传参方式&#xff1a;json格式、表单格式 首先了解下&#xff0c;postman进行接口测试&#xff0c;必须条件是&#xff1a; ♥请求地址 ♥请求协议 ♥请求方式 ♥请求头 ♥参数 json格式 先看一下接口文档&#xff0c;根据接口文档&…

算法通关村——如何使用中序和后序来恢复一棵二叉树

通过序列构造二叉树 给出以下三个二叉树遍历的序列&#xff1a; (1) 前序: 1 2 3 4 5 6 8 7 9 10 11 12 13 15 14 (2) 中序: 3 4 8 6 7 5 2 1 10 9 11 15 13 14 12 (3) 后序: 8 7 6 5 4 3 2 10 15 14 13 12 11 9 1 前中序复原二叉树 所需序列 (1) 前序: 1 2 3 4 5 6 8 7 9 10 …

UG\NX二次开发 属性更新的注意事项 UF_ATTR_assign

文章作者:里海 来源网站:https://blog.csdn.net/WangPaiFeiXingYuan 简介: 将属性值连接到表达式。如果原始部件中已经存在同名的常量值属性,则会导致属性连接到表达式失败,并且不返回错误值。如果原始部件同名的属性不存在,或者存在同名的连接表达式的属性,则会…

Django之JWT库与SimpleJWT库的使用

Django之JWT库与SimpleJWT库的使用 JWTJWT概述头部(header)载荷(payload)签名(signature) Django使用JWT说明jwt库的使用安装依赖库配置settings.py文件配置urls.py文件创建视图配置权限 SimpleJWT库的使用安装SimpleJWT库配置Django项目配置路由创建用户接口测试身份认证自定义…

小狐狸GPT付费创作系统WEB版源码-登录权限验证逻辑

小狐狸GPT付费创作系统WEB版默认是需要公众号关注登录&#xff0c;一直想改成账号密码登录形式&#xff0c;继续查看接口部分 获取系统设置信息的接口 /web.php/login/system 从header里取x-site作为sitecode&#xff0c;如果取不到默认1 从setting表里查出设置数据&#xff0c…

【动态规划part17】| 647.回文子串、516.最长回文子序列

&#x1f388;LeetCode647.回文子串 链接&#xff1a;647.回文子串 给你一个字符串 s &#xff0c;请你统计并返回这个字符串中 回文子串 的数目。 回文字符串 是正着读和倒过来读一样的字符串。 子字符串 是字符串中的由连续字符组成的一个序列。 具有不同开始位置或结束位置…

C++初阶函数重载

目录 函数重载函数名修饰规则 函数重载 C语言不允许同名函数 CPP可以&#xff0c;但是要求构成函数重载 函数名相同&#xff0c;参数不同(参数类型、参数个数、参数类型的顺序)&#xff0c;返回值不同不能构成重载 int Add(int left, int right) {cout << "int Ad…

【JS交互篇】BOM基础、Window、Location、Navagator、Screen、History对象

一、BOM 概述 在 JavaScript 语言中有三种对象&#xff1a;内置对象、宿主对象、自定义对象。 宿主对象就是执行 JavaScript 脚本的环境所提供的对象。对于网页编程来说&#xff0c;js 是运行在浏览器上的&#xff0c;所以对于网页编程来说&#xff0c;宿主对象就是浏览器对象…

“我要找到你”——主动救助,让乡村重疾儿童看病不再“两眼一抹黑”

走出医院&#xff0c;主动寻找 郑小伟是河北省阜平县妇幼保健院的儿科主任&#xff0c;在这里工作快20年了。从2021年11月开始&#xff0c;她的桌上一直摆着厚厚一叠阜平县重疾儿童住院名单。她对着上面的名字一个个打电话&#xff0c;手机打不通&#xff0c;就换座机再接着打…

高级语言相关理论[编译VS解释,动态VS静态,强类型VS弱类型]

编译型语言 VS 解释型语言 计算机高级语言按程序执行方式分为编译型和解释型 编译型和解释型语言的执行流程 编译型语言 所有源代码一次性通过编译器转换成二进制指令,即生成一个可执行程序(如Windows下的.exe),可执行程序包含的就是机器码;且无需重新编译,实现一次编译,无限…

Grafana集成prometheus(1.Prometheus安装)

下载docker镜像 docker pull prom/prometheus docker pull prom/node-exporter启动 node-exporter 该程序用以采集机器内存等数据 启动脚本 docker run -d -p 9100:9100 prom/node-exporter ss -anptl | grep 9100启动截图 prometheus 启动脚本 # 3b907f5313b7 为镜像i…

工业电子中的安森美推出PWM控制器 NCP1252ADR2G 用于正向和反激应用

NCP1252ADR2G 深力科 控制器提供了构建专用于 ATX 电源和任何正向应用的成本高效且可靠的 AC-DC 开关电源所需的一切内容。由于使用内部固定的计时器&#xff0c;可以在不依赖辅助 Vcc 的情况下检测输出过载。欠压输入针对低输入电压提供保护&#xff0c;提高了转换器安全性。最…

c语言——杨辉三角

//杨辉三角 #include<stdio.h> int main() {int i,j,k,n0,a[10][10];while(n<0||n>13){/*行数不超过13&#xff0c;为了显示规范*/printf("n即输入行数");scanf("%d",&n);}printf("%d行杨辉三角如下&#xff1a;\n",n);for(i1;i…

迁移学习:使用Restnet预训练模型构建高效的水果识别模型

目录 引言 1 迁移学习 1.1 什么是迁移学习 1.2 迁移学习能解决什么问题 1.3 迁移学习面临的三个问题 1.3.1 何时迁移 1.3.2 何处迁移 1.3.3 如何迁移 1.4 迁移学习的分类 1.4.1 按照学习方式的划分 1.4.2 按照使用方法的划分 2 Restnet网络 2.1 Restnet介绍 2.2 Re…

【FAQ】EasyGBS平台通道显示在线,视频无法播放并报错400的排查

EasyGBS是基于国标GB28181协议的视频云服务平台&#xff0c;它可以支持国标协议的设备接入&#xff0c;在视频能力上能实现直播、录像存储、检索与回放、云台控制、告警上报、语音对讲、平台级联等功能&#xff0c;既能作为业务平台使用&#xff0c;也能作为能力层平台调用。 我…

sentinel组件

目录 定义 4.加SentinelResource,blockHander是超过阈值之后执行的函数 5.设置阈值 6.springboot集成sentinel 定义 1.sentinel知道当前流量大小&#xff0c;在浏览器和后端之间加sentinel控制流量&#xff0c;避免大批量的瞬时请求都达到服务上&#xff0c;将服务压垮 2.…

数学建模-爬虫系统学习

尚硅谷Python爬虫教程小白零基础速通&#xff08;含python基础爬虫案例&#xff09; 内容包括&#xff1a;Python基础、Urllib、解析&#xff08;xpath、jsonpath、beautiful&#xff09;、requests、selenium、Scrapy框架 python基础 进阶&#xff08;字符串 列表 元组 字典…

Ceph入门到精通-远程开发Windows下使用SSH密钥实现免密登陆Linux服务器

工具&#xff1a; win10、WinSCP 服务器生成ssh密钥&#xff1a; 打开终端&#xff0c;使账号密码登录&#xff0c;输入命令 ssh-keygen -t rsa Winscp下载 Downloading WinSCP-6.1.1-Setup.exe :: WinSCP window 生成密钥 打开powershell ssh-keygen -t rsa 注意路径 …