《算法导论》学习笔记之Chapter8线性时间排序

2024-05-16 00:18

本文主要是介绍《算法导论》学习笔记之Chapter8线性时间排序,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

第8章 线性时间排序

前面介绍的包括归并排序,堆排序和快速排序,最后的次序都依赖于元素之间的比较,叫做比较排序

归并排序和堆排序都是渐近最优的,且任何已知的比较排序最多就是在常数因子上优于他们。即比较排序的时间复杂度下界就是Ω(nlogn)。

线性时间排序算法,包括:基数排序,计数排序和桶排序,是靠运算不是比较来排序的,下界Ω(nlogn)不是他们的下界。


计数排序:假设n个输入元素中的每一个都是在0-k区间内的一个整数,其中k为某个整数。当k=O(n)时,排序的运行时间为O(n)。基本思想是:对每一个输入元素x,确定小于x的元素个数。利用这一信息,直接把x放到他在输出数组中中的位置上了。当出现元素相同时,需要稍作修改。代码如下:

        //计数排序
	public void CountingSort(int[] a, int[] b, int k){
		int[] c = new int[k];
		for (int i = 0; i < k; i++){
			c[i] = 0;
		}
		//统计每个位置上的数字都分别有几个
		for (int j = 0; j < a.length; j++){
			c[a[j]] = c[a[j]] + 1;
		}
		//统计小于等于该排位上的数字的总数
		for(int i = 1; i < k; i++){
			c[i] = c[i] + c[i - 1];
		}
		//根据c上面每个排位的数字把a中的数字分别直接放到其排序后的数字上
		for (int j = a.length - 1; j >= 0; j--){
			b[c[a[j]] - 1] = a[j];
			c[a[j]] = c[a[j]] - 1;
		}
	}

计数排序的下界是Ω(nlogn),且是稳定的。具有相同值的元素在输出数组中的相对次序与输入数组中的相对次序相同。


基数排序:是一种用在卡片排序机上的算法。


桶排序:假设输入数据服从均匀分布,平均情况下他的时间代价为O(n)。与计数排序类似,桶排序也是对输入数据作出某种假设,所以速度较快。桶排序假设输入数据是由一个随机过程产生,该过程将元素均匀、独立地分布在[0,1)区间上。桶排序将[0,1)区间划分为n个相同大小的子区间,或称为桶。排序思想是:因为输入数据是均匀、独立地分布在[0,1)区间上,所以一般不会出现很多数落在同一个桶中的情况。为了的到输出结果,我们先对每个桶中的数进行排序,然后遍历每个桶,按照次序把各个桶的元素列出来即可。代码如下:

         public void BucketSort(Double[] a) {int n = a.length;/*** 创建链表(桶)集合并初始化,集合中的链表用于存放相应的元素*/int bucketNum = 10; // 桶数LinkedList<LinkedList<Double>> buckets = new LinkedList<LinkedList<Double>>();for (int i = 0; i < bucketNum; i++) {LinkedList<Double> bucket = new LinkedList<Double>();buckets.add(bucket);}// 把元素放进相应的桶中for (int i = 0; i < n; i++) {int index = (int) (a[i] * bucketNum);buckets.get(index).add(a[i]);}// 对每个桶中的元素排序,并放进a中int index = 0;for (LinkedList<Double> linkedList : buckets) {int size = linkedList.size();if (size == 0) {continue;}// 把LinkedList<Double>转化为Double[]的原因是,之前已经实现了对数组进行排序的算法Double[] temp = new Double[size];for (int i = 0; i < temp.length; i++) {temp[i] = linkedList.get(i);}// 利用插入排序对temp排序InsertSort(temp);for (int i = 0; i < temp.length; i++) {a[index] = temp[i];index++;}}}

只要输入数据满足以下条件:所有桶的大小的平方和与总的元素数呈线性关系,那么桶排序就可以按照线性时间 θ(n)完成排序。


这篇关于《算法导论》学习笔记之Chapter8线性时间排序的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/993330

相关文章

Feign Client超时时间设置不生效的解决方法

《FeignClient超时时间设置不生效的解决方法》这篇文章主要为大家详细介绍了FeignClient超时时间设置不生效的原因与解决方法,具有一定的的参考价值,希望对大家有一定的帮助... 在使用Feign Client时,可以通过两种方式来设置超时时间:1.针对整个Feign Client设置超时时间

springboot+dubbo实现时间轮算法

《springboot+dubbo实现时间轮算法》时间轮是一种高效利用线程资源进行批量化调度的算法,本文主要介绍了springboot+dubbo实现时间轮算法,文中通过示例代码介绍的非常详细,对大家... 目录前言一、参数说明二、具体实现1、HashedwheelTimer2、createWheel3、n

Mybatis 传参与排序模糊查询功能实现

《Mybatis传参与排序模糊查询功能实现》:本文主要介绍Mybatis传参与排序模糊查询功能实现,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、#{ }和${ }传参的区别二、排序三、like查询四、数据库连接池五、mysql 开发企业规范一、#{ }和${ }传参的

Java实现时间与字符串互相转换详解

《Java实现时间与字符串互相转换详解》这篇文章主要为大家详细介绍了Java中实现时间与字符串互相转换的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、日期格式化为字符串(一)使用预定义格式(二)自定义格式二、字符串解析为日期(一)解析ISO格式字符串(二)解析自定义

SpringBoot实现MD5加盐算法的示例代码

《SpringBoot实现MD5加盐算法的示例代码》加盐算法是一种用于增强密码安全性的技术,本文主要介绍了SpringBoot实现MD5加盐算法的示例代码,文中通过示例代码介绍的非常详细,对大家的学习... 目录一、什么是加盐算法二、如何实现加盐算法2.1 加盐算法代码实现2.2 注册页面中进行密码加盐2.

Java时间轮调度算法的代码实现

《Java时间轮调度算法的代码实现》时间轮是一种高效的定时调度算法,主要用于管理延时任务或周期性任务,它通过一个环形数组(时间轮)和指针来实现,将大量定时任务分摊到固定的时间槽中,极大地降低了时间复杂... 目录1、简述2、时间轮的原理3. 时间轮的实现步骤3.1 定义时间槽3.2 定义时间轮3.3 使用时

Python如何获取域名的SSL证书信息和到期时间

《Python如何获取域名的SSL证书信息和到期时间》在当今互联网时代,SSL证书的重要性不言而喻,它不仅为用户提供了安全的连接,还能提高网站的搜索引擎排名,那我们怎么才能通过Python获取域名的S... 目录了解SSL证书的基本概念使用python库来抓取SSL证书信息安装必要的库编写获取SSL证书信息

C++快速排序超详细讲解

《C++快速排序超详细讲解》快速排序是一种高效的排序算法,通过分治法将数组划分为两部分,递归排序,直到整个数组有序,通过代码解析和示例,详细解释了快速排序的工作原理和实现过程,需要的朋友可以参考下... 目录一、快速排序原理二、快速排序标准代码三、代码解析四、使用while循环的快速排序1.代码代码1.由快

Java进阶学习之如何开启远程调式

《Java进阶学习之如何开启远程调式》Java开发中的远程调试是一项至关重要的技能,特别是在处理生产环境的问题或者协作开发时,:本文主要介绍Java进阶学习之如何开启远程调式的相关资料,需要的朋友... 目录概述Java远程调试的开启与底层原理开启Java远程调试底层原理JVM参数总结&nbsMbKKXJx

如何通过Golang的container/list实现LRU缓存算法

《如何通过Golang的container/list实现LRU缓存算法》文章介绍了Go语言中container/list包实现的双向链表,并探讨了如何使用链表实现LRU缓存,LRU缓存通过维护一个双向... 目录力扣:146. LRU 缓存主要结构 List 和 Element常用方法1. 初始化链表2.