十大排序算法之线性时间非比较类排序

线性时间非比较类排序

线性时间的算法执行效率也较高,从时间占用上看,线性时间非比较类排序要优于非线性时间排序,但其空间复杂度较非线性时间排序要大一些。因为线性时间非比较类排序算法会额外申请一定的空间进行分配排序,这也是它的典型特点——以空间换时间。而且,线性时间非比较类排序对待排序元素的要求较为严格,如计数排序要求待排序序列的差值范围不能太大,桶排序要求元素的分布要尽量均匀等。

线性时间非比较类排序的优势和局限性

线性时间算法的高效性

线性时间比较类排序算法具有较高的执行效率。相对于非线性时间排序算法,线性时间算法在时间占用上更为优越。它们能够在O(n)的时间复杂度内完成排序,这意味着算法的执行时间随着待排序元素数量的增加而线性增长。这使得线性时间算法成为处理大规模数据集的理想选择,能够快速有效地完成排序任务。

以空间换时间的策略

线性时间比较类排序算法常常以空间换时间的策略来提升排序性能。这些算法会额外申请一定的空间来进行元素分配和排序操作。例如,计数排序和桶排序都需要额外的空间用于分配元素。虽然这增加了空间复杂度,但却能够在时间复杂度上获得显著的提升。以空间换时间的策略使得线性时间比较类排序算法在某些场景下非常适用。

限制和要求

线性时间比较类排序算法对待排序元素有一定的限制和要求。例如,计数排序要求待排序序列的差值范围不能太大,否则会导致额外的空间消耗。桶排序则要求元素的分布尽量均匀,以充分利用桶的优势。因此,在选择线性时间比较类排序算法时,需要根据待排序数据的特点来确定算法的适用性。

例子

在Java编程语言中,实现线性时间比较类排序算法可以帮助我们更直观地理解其工作原理和优势。以下以基数排序(Radix Sort)为例,这是一种线性时间复杂度的非比较类排序算法,适用于整数排序且对数值范围有一定要求。

public class RadixSort {// 基数排序函数public static void radixsort(int[] arr) {if (arr == null || arr.length == 0)return;int max = Arrays.stream(arr).max().getAsInt(); // 找出数组中的最大值for (int exp = 1; max / exp > 0; exp *= 10) { // 按照每一位进行计数排序countingSort(arr, exp);}}// 计数排序辅助函数private static void countingSort(int[] arr, int exp) {int n = arr.length;int[] output = new int[n]; // 输出数组int[] count = new int[10]; // 计数数组Arrays.fill(count, 0);// 计算每个位上出现的次数for (int i = 0; i < n; i++) {count[(arr[i] / exp) % 10]++;}// 将计数数组转换为前缀和,便于计算输出位置for (int i = 1; i < 10; i++) {count[i] += count[i - 1];}// 根据计数数组将元素放到正确的位置for (int i = n - 1; i >= 0; i--) {output[count[(arr[i] / exp) % 10] - 1] = arr[i];count[(arr[i] / exp) % 10]--;}// 将临时数组复制回原数组System.arraycopy(output, 0, arr, 0, n);}// 测试代码public static void main(String[] args) {int[] array = {170, 45, 75, 90, 802, 24, 2, 66};radixsort(array);System.out.println(Arrays.toString(array));}
}

上面这段Java代码展示了基数排序的基本实现,它首先找到数组中的最大值来确定需要处理的位数,然后逐个按照每一位进行计数排序。虽然基数排序的空间复杂度相对较高,但它能在线性时间内完成排序,尤其对于大规模整数排序场景具有很高的实用价值。

除了基数排序,还有一种线性时间非比较类排序算法——桶排序(Bucket Sort),它适用于待排序元素在一定范围内且分布均匀的情况。桶排序的思想是将数组中的元素分到有限数量的“桶”中,然后对每个桶分别进行排序,最后按顺序合并所有桶中的元素。

以下是Java实现桶排序的基本示例:

public class BucketSort {// 桶排序函数public static void bucketSort(int[] arr, int bucketSize) {if (arr == null || arr.length == 0)return;int minValue = Arrays.stream(arr).min().getAsInt();int maxValue = Arrays.stream(arr).max().getAsInt();// 创建桶List<List<Integer>> buckets = new ArrayList<>();for (int i = 0; i <= maxValue - minValue; i += bucketSize) {buckets.add(new ArrayList<>());}// 将元素分配到各个桶中for (int num : arr) {int index = (num - minValue) / bucketSize;buckets.get(index).add(num);}// 对每个桶进行排序,这里使用插入排序作为子排序算法for (List<Integer> bucket : buckets) {Collections.sort(bucket);}// 合并所有已排序的桶int index = 0;for (List<Integer> bucket : buckets) {for (Integer num : bucket) {arr[index++] = num;}}}// 测试代码public static void main(String[] args) {int[] array = {5, 3, 8, 1, 9, 6, 7, 2, 4};bucketSort(array, 3);System.out.println(Arrays.toString(array));}
}

在上面这段Java代码中,首先计算出数组的最小值和最大值以确定桶的数量,并初始化桶列表。接着遍历输入数组,根据元素值将其放入对应的桶中。之后对每个桶内部采用插入排序或其他适合的小规模排序算法进行排序。最后,按照桶的顺序合并所有已排序的桶,从而得到最终排序结果。
虽然桶排序在理想情况下可以达到线性时间复杂度,但其性能取决于输入数据的分布情况以及所选桶大小等因素。如果数据分布不均匀或者桶大小选择不当,可能会导致实际运行效率下降。

上面的两个例子会在后面的文章里详细讲解

总结

线性时间非比较类排序算法具有高效的执行效率和较低的时间复杂度,适用于处理大规模数据集的排序任务。它们以空间换时间的策略,在一定的限制和要求下,能够快速有效地完成排序操作。然而,需要根据待排序数据的特点来选择合适的算法,以充分发挥线性时间比较类排序算法的优势。

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

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

相关文章

百度Apollo荣获量子位-智能车参考“2023智能车年度评选”两大奖项

2023&#xff0c;是谁在引领智能车浪潮&#xff1f; 量子位-智能车参考发布“2023智能车年度评选”结果。经过业内专业评审和垂直社群万人票选后&#xff0c;百度Apollo荣获「年度十大智能车技术产品/方案」「年度十大智能车领军人物」两项大奖。 年度十大智能车技术产品\方…

jvm问题自查思路

本文聊一下最近处理了一些jvm的问题上&#xff0c;将这个排查和学习过程分享一下&#xff0c;看了很多资料&#xff0c;最终都会落地到几个工具的使用&#xff0c;本文主要是从文档学习、工具学习和第三方技术验证来打开认知和实践&#xff0c;希望有用。 一、文档 不仅知道了…

springboot+vue居民小区设备报修系统

小区报修系统可以提高设施维护的效率&#xff0c;减少机构的人力物力成本&#xff0c;并使得维修人员可以更好地了解维护设备的情况&#xff0c;及时解决问题。 对于用户来说&#xff0c;报修系统也方便用户的维修请求和沟通&#xff0c;提高了用户的满意度和信任。其次小区报修…

imazing是啥?imazing有什么用

iMazing 是一款用于管理 iOS 设备的第三方软件。它支持在 Windows 或 Mac 电脑上对连接的 iPhone、iPad 或 iPod 进行数据传输、备份和管理工作。用户可以通过 iMazing 传输和备份包括照片、音乐、铃声、视频、电子书和通讯录等在内的多种信息。iMazing 的功能与苹果官方的 iTu…

函数及函数的定义

前言&#xff1a; 在之前介绍指针的时候&#xff0c;小编发现有些地方需要用函数&#xff0c;所以小编决定先带领大家学习函数&#xff0c;然后再学习指针。 函数是从英文function翻译过来的&#xff0c;其实function在英文中的意思就是函数&#xff0c;也是功能的意思&#xf…

UE4运用C++和框架开发坦克大战教程笔记(十九)(第58~60集)完结

UE4运用C和框架开发坦克大战教程笔记&#xff08;十九&#xff09;&#xff08;第58~60集&#xff09;完结 58. 弹窗显示与隐藏59. UI 面板销毁60. 框架完成与总结 58. 弹窗显示与隐藏 这节课我们先来补全 TransferMask() 里对于 Overlay 布局类型面板的遮罩转移逻辑&#xff…

C++,stl,栈stack和队列queue详解

1.栈stack 1.stack基本概念 2.stack常用接口 代码示例&#xff1a; #include<bits/stdc.h> using namespace std;int main() {stack<int> stk;stk.push(7);stk.push(9);stk.push(5);cout << "栈的size为&#xff1a;" << stk.size() <…

ensp命令大全详解

以下是常用的Cisco模拟器&#xff08;如Packet Tracer&#xff09;中的命令大全详解&#xff1a; 1. 基础命令 - enable&#xff1a;进入特权模式&#xff08;enable mode&#xff09;。 - configure terminal&#xff1a;进入全局配置模式&#xff08;global configuration …

B2077 角谷猜想(洛谷)

题目描述 所谓角谷猜想&#xff0c;是指对于任意一个正整数&#xff0c;如果是奇数&#xff0c;则乘 33 加 11&#xff0c;如果是偶数&#xff0c;则除以 22&#xff0c;得到的结果再按照上述规则重复处理&#xff0c;最终总能够得到 11。如&#xff0c;假定初始整数为 55&…

【STL】list模拟实现

vector模拟实现 一、接口大框架函数声明速览二、结点类的模拟实现1、构造函数 三、迭代器类的模拟实现1、迭代器类存在的意义2、迭代器类的模板参数说明3、构造函数4、运算符的重载&#xff08;前置和后置&#xff09;&#xff08;1&#xff09;前置&#xff08;2&#xff09;后…

十、项目开发总结报告(软件工程)

1&#xff0e;引言 1.1编写目的 1.2项目背景 1.3定义 1.4参考资料 2&#xff0e;开发结果 2.1产品 2.2主要功能及性能 2.3所用工时 2.4所用机时 2.5进度 2.6费用 3&#xff0e;评价 3.1生产率评价 3.2技术方案评价 3.3产品质量评价 4&#xff…

C语言数组名涵义

1. 数组名涵义 在C语言程序中&#xff0c;数组的出现有两种可能的含义&#xff1a; 代表整个数组代表其首元素的地址 当出现以下情形时&#xff0c;数组代表的是整个数组&#xff1a; 在数组定义中在 sizeof 运算表达式中在取址符&中 当出现其他情形时&#xff0c;数组代表…

office文件转pdf在线预览

一、工具类 package com.sby.utils;import java.io.File; import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.InputStream; import java.math.RoundingMode; import java.text.DecimalFormat; import java.util.Locale;import com.aspose.cel…

【Web】vulhub Shiro-550反序列化漏洞复现学习笔记

目录 Shiro简介 复现流程 工具一把梭 半脚本半手动 原理分析 反序列化入口 常见的key 登录过程 验证过程 利用原理 Shiro简介 Apache Shiro 是一个强大且易于使用的 Java 安全框架&#xff0c;用于身份验证、授权、加密和会话管理等安全功能。Shiro 的设计目标是简单…

【Spring】springmvc如何处理接受http请求

目录 ​编辑 1. 背景 2. web项目和非web项目 3. 环境准备 4. 分析链路 5. 总结 1. 背景 今天开了一篇文章“SpringMVC是如何将不同的Request路由到不同Controller中的&#xff1f;”&#xff1b;看完之后突然想到&#xff0c;在请求走到mvc 之前服务是怎么知道有请求进来…

大模型是如何实现Function Call函数调用的?

▼最近直播超级多&#xff0c;预约保你有收获 近期直播&#xff1a;《Agent 企业级应用案例实战》 —1— 大模型如何实现函数调用&#xff1f; 大模型要实现精确的函数调用&#xff08;Function Call&#xff09;需要理解能力和逻辑能力&#xff0c;理解能力就是对用户的 Prom…

redhat grub.cfg配置文件丢失或报错解决

1.实验环境&#xff1a;把grub.cfg删除 [rootexample ~]# rm -rf /boot/grub2/grub.cfg 2.重启服务器 3&#xff0c;发现进入系统失败 输入以下命令 ls: 列出当前设备上的文件和目录。 grub> ls (hd0) (hd0,msdos3) (hd0,msd0s2) (hd0,msdos1) #一般第一个为/boot分区se…

技术精英求职必备:Java开发工程师简历制作全指南

投简历找工作嘛&#xff0c;这事儿其实就跟相亲差不多&#xff0c;得让对方一眼就看上你。 在这场职场的‘相亲’中&#xff0c;怎样才能让你的简历脱颖而出&#xff0c;成为HR眼中的理想‘对象’呢&#xff1f;来&#xff0c;我给你支几招&#xff0c;让你的简历更吸引人。 …

HiveSQL——用户中两人一定认识的组合数

注&#xff1a;参考文章&#xff1a; SQL之用户中两人一定认识的组合数--HQL面试题36【快手数仓面试题】_sql面试题-快手-CSDN博客文章浏览阅读1.2k次&#xff0c;点赞3次&#xff0c;收藏12次。目录0 需求分析1 数据准备2 数据分析3 小结0 需求分析设表名&#xff1a;table0现…

MPLS VPN功能组件(3)

私网标签分配 通过MPBGP为VPNv4路由分配内层标签 PE从CE接收到IPv4路由后&#xff0c;对该路由加上相应VRF的RD&#xff08;RD手动配置&#xff09;&#xff0c;使其成为一条VPNV4路由&#xff0c;然后在路由通告中更改下一跳属性为自己&#xff0c;通常是自己的Loopback地址…