《算法导论》学习笔记之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

相关文章

如何利用Java获取当天的开始和结束时间

《如何利用Java获取当天的开始和结束时间》:本文主要介绍如何使用Java8的LocalDate和LocalDateTime类获取指定日期的开始和结束时间,展示了如何通过这些类进行日期和时间的处... 目录前言1. Java日期时间API概述2. 获取当天的开始和结束时间代码解析运行结果3. 总结前言在J

Java深度学习库DJL实现Python的NumPy方式

《Java深度学习库DJL实现Python的NumPy方式》本文介绍了DJL库的背景和基本功能,包括NDArray的创建、数学运算、数据获取和设置等,同时,还展示了如何使用NDArray进行数据预处理... 目录1 NDArray 的背景介绍1.1 架构2 JavaDJL使用2.1 安装DJL2.2 基本操

Spring排序机制之接口与注解的使用方法

《Spring排序机制之接口与注解的使用方法》本文介绍了Spring中多种排序机制,包括Ordered接口、PriorityOrdered接口、@Order注解和@Priority注解,提供了详细示例... 目录一、Spring 排序的需求场景二、Spring 中的排序机制1、Ordered 接口2、Pri

修改若依框架Token的过期时间问题

《修改若依框架Token的过期时间问题》本文介绍了如何修改若依框架中Token的过期时间,通过修改`application.yml`文件中的配置来实现,默认单位为分钟,希望此经验对大家有所帮助,也欢迎... 目录修改若依框架Token的过期时间修改Token的过期时间关闭Token的过期时js间总结修改若依

Go Mongox轻松实现MongoDB的时间字段自动填充

《GoMongox轻松实现MongoDB的时间字段自动填充》这篇文章主要为大家详细介绍了Go语言如何使用mongox库,在插入和更新数据时自动填充时间字段,从而提升开发效率并减少重复代码,需要的可以... 目录前言时间字段填充规则Mongox 的安装使用 Mongox 进行插入操作使用 Mongox 进行更

对postgresql日期和时间的比较

《对postgresql日期和时间的比较》文章介绍了在数据库中处理日期和时间类型时的一些注意事项,包括如何将字符串转换为日期或时间类型,以及在比较时自动转换的情况,作者建议在使用数据库时,根据具体情况... 目录PostgreSQL日期和时间比较DB里保存到时分秒,需要和年月日比较db里存储date或者ti

大数据小内存排序问题如何巧妙解决

《大数据小内存排序问题如何巧妙解决》文章介绍了大数据小内存排序的三种方法:数据库排序、分治法和位图法,数据库排序简单但速度慢,对设备要求高;分治法高效但实现复杂;位图法可读性差,但存储空间受限... 目录三种方法:方法概要数据库排序(http://www.chinasem.cn对数据库设备要求较高)分治法(常

Python中的随机森林算法与实战

《Python中的随机森林算法与实战》本文详细介绍了随机森林算法,包括其原理、实现步骤、分类和回归案例,并讨论了其优点和缺点,通过面向对象编程实现了一个简单的随机森林模型,并应用于鸢尾花分类和波士顿房... 目录1、随机森林算法概述2、随机森林的原理3、实现步骤4、分类案例:使用随机森林预测鸢尾花品种4.1

Python中lambda排序的六种方法

《Python中lambda排序的六种方法》本文主要介绍了Python中使用lambda函数进行排序的六种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们... 目录1.对单个变量进行排序2. 对多个变量进行排序3. 降序排列4. 单独降序1.对单个变量进行排序

Python 标准库time时间的访问和转换问题小结

《Python标准库time时间的访问和转换问题小结》time模块为Python提供了处理时间和日期的多种功能,适用于多种与时间相关的场景,包括获取当前时间、格式化时间、暂停程序执行、计算程序运行时... 目录模块介绍使用场景主要类主要函数 - time()- sleep()- localtime()- g