Java常见的排序算法

排序分为内部排序和外部排序(外部存储)

常见的七大排序,这些都是内部排序

在这里插入图片描述

1、插入排序:直接插入排序

1、插入排序:每次将一个待排序的记录,按其关键字的大小插入到前面已排序好的记录序列 中的适当位置,直到全部记录插入完成为止。稳定排序算法

一个排序算法的稳定性与不稳定性是通过排序后相同元素的先后顺序 来判断的。

稳定性:如果排序前后,具有相同关键字的元素的相对顺序没有改变,则排序算法被认为是稳定的。

不稳定性:如果排序前后,具有相同关键字的元素的相对顺序发生了改变,则排序算法被认为是不稳定的。

  • 从第一个元素开始,该元素可以认为已经被排序
  • 取出下一个元素,在已经排序的元素序列中从后往前扫描
  • 如果该元素(已排序)大于新元素,将该元素移到下一位置;
  • 直到找到已排序的元素小于或者等于新元素的位置
  • 将新元素插入到该位置后(这就保证了相同元素的顺序和排序前一样,所以是稳定排序
  • 重复扫描

2、代码

public class InsertSort {
/**
* <>
* @method: insertSort
* @Param: [arr]
* @Return: void
* @exception:
* @Author: fsy
* @Date: 19-5-29 下午3:50
* @description:
*
*/
public void insertSort(int[] arr){//需要插入的数int insertNum;for (int i=1; i <arr.length ; i++) {insertNum=arr[i];//序列元素个数int j=i-1;//将大于insertNum的元素往后移动while (j>=0&&arr[j]>insertNum){arr[j+1]=arr[j];j--;}//找到位置,插入当前元素arr[j+1]=insertNum;}}

3、复杂度分析

时间复杂度:

  • O(n²)

空间复杂度:

  • O(1)

4、总结:

插入排序所需的时间取决于输入元素的初始顺序 。对于一个很大且其中的元素已经有序(或接近有序)的数组进行排序将会比随机顺序的数据或是逆序数据进行排序要快的多

2、插入排序:希尔排序

1、希尔排序:希尔排序的本质就是分组插入排序 ,又称缩小增量法。将整个无序序列分割成若干个子序列(由相隔某个“增量”的元素组成)分别进行直接插入排序,然后依次缩减增量再进行排序,待整个序列中的元素基本有序时,再对全体元素进行一次直接插入排序 。因为进行直接插入排序时元素基本有序,所以效率是很高的,因此希尔排序在时间效率上有很大提高。希尔排序是不稳定排序算法

2、代码

public void shellSort(int[] d) { //d[]为增量数组RecordNode temp;int i, j;System.out.println("希尔排序");//控制增量,增量减半,若干趟扫描for (int k = 0; k < d.length; k++) {//一趟中若干子表,每个记录在自己所属子表内进行直接插入排序int dk = d[k];for (i = dk; i < this.curlen; i++) {temp = r[i];for (j = i - dk; j >= 0 && temp.key.compareTo(r[j].key) < 0; j -= dk) {r[j + dk] = r[j];}r[j + dk] = temp;}System.out.print("增量dk=" + dk + "  ");}
}

3、复杂度分析

时间复杂度:

  • O(nlog2 n)

空间复杂度:

  • O(1)

4、总结

希尔排序更高效是因为它权衡了子数组的规模和有序性。排序之初,各个子数组都很短,排序之后子数组都是部分有序的,这两种情况都很适合插入排序

3、选择排序:简单选择排序

1、选择排序:工作原理如下。首先在未排序序列中找到最小(大)元素,存放在排序序列的初始位置。然后,再从剩余未排序元素中继续寻找最小(大)元素,然后放到已排序列的末尾,以此类推,直到所有元素排序完毕。选择排序的主要优点是与数据移动有关,如果某个元素位于正确的最终位置上,则它不会被移动。选择排序每次交换一对元素,它们当中至少有一个将被移到其最终位置上,因此对n个元素的表进行排序最多n-1次交换。不稳定排序

2、代码:

public static void sort(int[] a) {for (int i = 0; i < a.length; i++) {int min = i;//选出之后待排序中值最小的位置for (int j = i + 1; j < a.length; j++) {if (a[j] < a[min]) {min = j;}}//最小值不等于当前值时进行交换if (min != i) {int temp = a[i];a[i] = a[min];a[min] = temp;}}
}

3、复杂度分析

时间复杂度:

  • O(n²)

空间复杂度:

  • O(1)

4、总结

选择排序非常简单和直观,但是也非常慢。无论是哪种情况,哪怕原数组已经排序完成,它也将花费将近n²/2次遍历来确认一遍。它的排序结果也是不稳定的。不耗费额外的内存空间

4、选择排序:堆排序

1、堆的定义如下:n个元素的序列(k1,k2,… ,kn)

当且仅当满足下列关系时,称之为堆

在这里插入图片描述

把此序列对应的二维数组看成是一个完全二叉树,那么堆的含义就是:完全二叉树中任何一个非叶子节点的值均不大于(或不小于)其左、右孩子节点的值 。由上述性质可知,大顶堆的堆顶的关键字肯定是所有关键字中最大的,小顶堆的堆顶的关键字是所有关键字中最小的。因此我们可以使用大顶堆进行升序排序,使用小顶堆进行降序排序。不稳定排序

基本思想:以大顶堆为例,堆排序的过程就是将待排序的序列构造成一个堆,选出堆中最大的移走,再把剩余的元素调整成堆,找出最大的再移走,重复直至有序

2、代码

/*** @param a*/
public static void sort(int[] a) {for (int i = a.length - 1; i > 0; i--) {max_heapify(a, i);//堆顶元素(第一个元素)与Kn交换int temp = a[0];a[0] = a[i];a[i] = temp;}
}/*****  将数组堆化*  i = 第一个非叶子节点。*  从第一个非叶子节点开始即可。无需从最后一个叶子节点开始。*  叶子节点可以看作已符合堆要求的节点,根节点就是它自己且自己以下值为最大。** @param a* @param n*/
public static void max_heapify(int[] a, int n) {int child;for (int i = (n - 1) / 2; i >= 0; i--) {//左子节点位置child = 2 * i + 1;//右子节点存在且大于左子节点,child变成右子节点if (child != n && a[child] < a[child + 1]) {child++;}//交换父节点与左右子节点中的最大值if (a[i] < a[child]) {int temp = a[i];a[i] = a[child];a[child] = temp;}}
}

3、复杂度分析

时间复杂度:O(nlog₂ n)

空间复杂度:O(1)

4、总结

由于堆排序中初始化堆的过程比较次数较多,因此不太使用于小序列 。由于多次任意下标相互交换位置,相同元素之间原本相对的顺序被破坏了,是不稳定排序。

5、交换排序:冒泡排序

1、冒泡排序:是一种简单排序。重复地走访过要排序的元素,一次比较两个元素,如果他们的顺序错误就把他们交换过来,走访数列的工作是重复地进行直到没有再需要交换,也就是说该数列已经排序完成。只在顺序不符合大小要求时交换,所以不会破坏相同元素间的顺序,因此是稳定排序

  • 比较相邻的元素,如果第一个比第二个大,就交换他们两个
  • 对每一对相邻元素做同样的工作,从开始第一对到结尾的最后一对,这步做完,最后的元素会是最大的数
  • 针对所有的元素重复以上的步骤,除了最后一个
  • 持续每次对越来越少的元素重复上面的步骤,直到没有任何一对数字需要比较

2、代码

public static void sort(int[] a) {//外层循环控制比较的次数for (int i = 0; i < a.length - 1; i++) {//内层循环控制到达位置for (int j = 0; j < a.length - i - 1; j++) {//前面的元素比后面大就交换if (a[j] > a[j + 1]) {int temp = a[j];a[j] = a[j + 1];a[j + 1] = temp;}}}
}

3、复杂度分析

时间复杂度:

  • 最好:O(n)
  • 最坏:O(n²)
  • 平均时间复杂度:O(n²)

空间复杂度:O(1)

4、总结:

是稳定的排序算法。

6、交换排序:快速排序

1、快速排序:使用分治法策略来把一个串行分为两个子串行。通过一趟排序将要排序的数据分割成独立的两部分,其中一部分的所有数据都比另一部分的所有数据都要小,然后再按此方法对这两部分数据分别进行快速排序,整个排序过程可以递归进行,以此达到整个数据变成有序序列。不稳定排序算法

2、代码

public static void sortByStack(int[] a) {Stack<Integer> stack = new Stack<Integer>();//初始状态的左右指针入栈stack.push(0);stack.push(a.length - 1);while (!stack.isEmpty()) {//出栈进行划分int high = stack.pop();int low = stack.pop();int pivotIndex = partition(a, low, high);//保存中间变量if (pivotIndex > low) {stack.push(low);stack.push(pivotIndex - 1);}if (pivotIndex < high && pivotIndex >= 0) {stack.push(pivotIndex + 1);stack.push(high);}}
}private static int partition(int[] a, int low, int high) {if (low >= high) return -1;int left = low;int right = high;//保存基准的值int pivot = a[left];while (left < right) {//从后向前找到比基准小的元素,插入到基准位置中while (left < right && a[right] >= pivot) {right--;}a[left] = a[right];//从前往后找到比基准大的元素while (left < right && a[left] <= pivot) {left++;}a[right] = a[left];}//放置基准值,准备分治递归快排a[left] = pivot;return left;
}

3、复杂度分析

时间复杂度:

  • 最好:O(nlog₂ n)
  • 最坏:O(n²)
  • 平均时间复杂度:O(nlog₂ n)

空间复杂度:O(1)

7、归并排序

1、归并排序:归并排序算法是将两个(或两个以上)有序表合并成一个新的有序表,即把待排序序列分为若干个子序列,每个子序列是有序的,然后再把有序子序列合并为整体有序序列。稳定排序算法

归并排序可通过两种方式实现:

  • 自上而下的递归
  • 自下而上的迭代

2、(递归的方法)代码

public class Merge {//归并所需的辅助数组private static int[] aux;public static void sort(int[] a) {//一次性分配空间aux = new int[a.length];sort(a, 0, a.length - 1);}public static void sort(int[] a, int low, int high) {if (low >= high) {return;}int mid = (low + high) / 2;//将左半边排序sort(a, low, mid);//将右半边排序sort(a, mid + 1, high);merge(a, low, mid, high);}/*** 该方法先将所有元素复制到aux[]中,然后在归并会a[]中。方法咋归并时(第二个for循环)* 进行了4个条件判断:* - 左半边用尽(取右半边的元素)* - 右半边用尽(取左半边的元素)* - 右半边的当前元素小于左半边的当前元素(取右半边的元素)* - 右半边的当前元素大于等于左半边的当前元素(取左半边的元素)*/public static void merge(int[] a, int low, int mid, int high) {//将a[low..mid]和a[mid+1..high]归并int i = low, j = mid + 1;for (int k = low; k <= high; k++) {aux[k] = a[k];}for (int k = low; k <= high; k++) {if (i > mid) {a[k] = aux[j++];} else if (j > high) {a[k] = aux[i++];} else if (aux[j] < aux[i]) {a[k] = aux[j++];} else {a[k] = aux[i++];}}}}

3、复杂度分析

时间复杂度:O(nlog₂n)

空间复杂度:O(n)

4、总结:主要缺点是所需的额外空间和N成正比(N为待排序数组的长度)

8、总结

各种排序性能对比:

排序类型平均情况最好情况最坏情况辅助空间稳定性
冒泡排序O(n²)O(n)O(n²)O(1)稳定
选择排序O(n²)O(n²)O(n²)O(1)不稳定
直接插入排序O(n²)O(n)O(n²)O(1)稳定
折半插入排序O(n²)O(n)O(n²)O(1)稳定
希尔排序O(nlog2 n)O(nlog2 n)O(nlog2 n)O(1)不稳定
归并排序O(nlog₂n)O(nlog₂n)O(nlog₂n)O(n)稳定
快速排序O(nlog₂n)O(nlog₂n)O(n²)O(nlog₂n)不稳定
堆排序O(nlog₂n)O(nlog₂n)O(nlog₂n)O(1)不稳定
计数排序O(n+k)O(n+k)O(n+k)O(k)稳定
桶排序O(n+k)O(n+k)O(n²)O(n+k)(不)稳定
基数排序O(d(n+k))O(d(n+k))O(d(n+kd))O(n+kd)稳定

稳定:冒泡排序、直接插入排序、归并排序

不稳定:选择排序、希尔排序、快速排序、堆排序

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

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

相关文章

ChatGPT AIGC 一个指令总结Python所有知识点

在ChatGPT中,直接输入一个指令就可以生成Python的所有知识点大纲。 非常实用的ChatGPT功能。 AIGC ChatGPT ,BI商业智能, 可视化Tableau, PowerBI, FineReport, 数据库Mysql Oracle, Office, Python ,ETL Excel 2021 实操,函数,图表,大屏可视化 案例实战 http://t.…

Angular中使用drag and drop实现文件拖拽上传,及flask后端接收

效果&#xff1a;拖拽文件到组件上面时 边框变大变红 松手后发送到服务器(或者点击蓝字手动选择文件)并且把文件名显示在框内&#xff0c;美化还没做 html <div class"drapBox"><div id"drop" (dragenter)"dragenter($event)" (dragov…

redis 应用 4: HyperLogLog

我们先思考一个常见的业务问题&#xff1a;如果你负责开发维护一个大型的网站&#xff0c;有一天老板找产品经理要网站每个网页每天的 UV 数据&#xff0c;然后让你来开发这个统计模块&#xff0c;你会如何实现&#xff1f; img 如果统计 PV 那非常好办&#xff0c;给每个网页一…

后台管理系统:项目路由搭建与品牌管理

路由的搭建 先删除一些不需要的界面 然后发现跑不起来&#xff0c;我们需要去配置 删减成这样&#xff0c;然后自己新建需要的路由组件 改成这样&#xff0c;这里要注意。我们是在layout这个大的组件下面的&#xff0c;meta 中的title就是我们侧边栏的标题&#xff0c;icon可…

Java eight 解读流(Stream)、文件(File)、IO和异常处理的使用方法

目录 Java 流(Stream)、文件(File)和IO读取控制台输入读写文件FileInputStreamFileOutputStream Java目录 Java 异常处理 Java 流(Stream)、文件(File)和IO java.io 包几乎包含了所有操作输入、输出需要的类。所有这些流类代表了输入源和输出目标。 Java.io 包中的流支持很多种…

51页企业数字化转型战略实践与启示PPT(附400份转型资料)

本资料来源公开网络&#xff0c;仅供个人学习&#xff0c;请勿商用&#xff0c;如有侵权请联系删除&#xff0c;更多内容浏览公众号&#xff1a;智慧方案文库 企业数字化转型之路.pptx企业数字化转型大数据湖一体化平台项目建设方案PPT.pptx企业数字化转型大数据湖一体化运营管…

Angular安全专辑之四 —— 避免服务端可能的资源耗尽(NodeJS)

express-rate-limit是一个简单实用的npm包,用于在Express应用程序中实现速率限制。它可以帮助防止DDoS攻击和暴力破解,同时还允许对API端点进行流控。 express-rate-limit及其主要功能 express-rate-limit是Express框架的一个流行中间件,它允许根据IP地址或其他标准轻松地对请求…

U盘文件恢复软件推荐,这几款高效恢复数据!

“我真的可以算得上是一个u盘杀手了&#xff0c;好多资料保存在u盘中&#xff0c;但经常都会由于粗心导致u盘中的数据丢失。大家有什么u盘文件恢复软件可以推荐吗&#xff1f;救救我的u盘吧&#xff01;” 在现代社会&#xff0c;人手一个u盘一点也不夸张。尤其是学生党和打工人…

阿里云架构

负载均衡slb 分类以及应用场景 负载均衡slb clb 传统的负载均衡(原slb) 支持4层和7层(仅支持对uri(location),域名进行转发) 一般使用slb(clb) alb 应用负载均衡 只支持7层,整合了nginx负载均衡的各种功能,可以根据用户请求头,响应头 如果需要详细处理用户请求(浏…

华为数通方向HCIP-DataCom H12-821题库(单选题:141-160)

第141题 Router-LSA 能够描述不同的链路类型&#xff0c;不属于Router LSA 链路类型的是以下哪一项? A、Link Type 可以用来描述到末梢网络的连接&#xff0c;即 SubNet B、Link Type 可以用来描述到中转网络的连接&#xff0c;即 TranNet C、Link Type 可以用来描述到另一…

2023京东口腔护理赛道行业数据分析(京东销售数据分析)

近年来&#xff0c;口腔护理逐渐成为年轻人重视的健康领域&#xff0c;从口腔护理整体市场来看&#xff0c;牙膏和牙刷等基础口腔护理产品仍占据主导地位。不过&#xff0c;随着口腔护理市场逐步朝向精致化、专业化、多元化等方向发展&#xff0c;不少新兴口腔护理产品受到消费…

算法通关村第8关【白银】| 二叉树的深度和高度问题

1.最大深度问题 思路&#xff1a;递归三部曲 第一步&#xff1a;确定参数和返回值 题目要求求二叉树的深度&#xff0c;也就是有多少层&#xff0c;需要传递一个root从底层向上统计 int maxDepth(TreeNode root) 第二步&#xff1a;确定终止条件 当递归到null时就说明到底了…

SpringMVC中Controller层获取前端请求参数的几种方式

SpringMVC中Controller层获取前端请求参数的几种方式 1、SpringMVC自动绑定2、使用RequestParam 注解进行接收3、RequestBody注解&#xff08;1&#xff09; 使用实体来接收JSON&#xff08;2&#xff09;使用 Map 集合接收JSON&#xff08;3&#xff09; 使用 List集合接收JSO…

持续性能优化:确保应用保持高性能

在当今数字化时代&#xff0c;应用程序的性能已经成为用户体验和业务成功的关键因素之一。无论是Web应用、移动应用还是企业级软件&#xff0c;用户对于速度和响应性的要求越来越高。因此&#xff0c;持续性能优化已经成为保证应用在竞争激烈的市场中脱颖而出的重要策略。 什么…

线性代数的学习和整理15:线性代数的快速方法

1 数学/线性代数里&#xff0c;其实很多东西的求得都有多种解决办法 很多概念&#xff0c;界定狠清晰&#xff0c;但是不好求 多种方法&#xff0c;拓宽思维 方法1&#xff1a;按定义直接去求解 方法2&#xff1a;按 2 比如求逆矩阵 概念方法&#xff0c;线性变化 增广矩阵…

JS 常见的 6 种继承方式

原型链继承 原型链继承是比较常见的继承方式之一&#xff0c;其中涉及的构造函数、原型和实例&#xff0c;三者之间存在着一定的关系&#xff0c;即每一个构造函数都有一个原型对象&#xff0c;原型对象又包含一个指向构造函数的指针&#xff0c;而实例则包含一个原型对象的指…

【遮天】李小曼回归,新形象无差云曦,短板竟是身材?

Hello,小伙伴们&#xff0c;我是小郑继续为大家深度解析遮天 最新一集《遮天》已经更新&#xff0c;在成功卖掉段德之后&#xff0c;叶凡便离开妖帝坟冢&#xff0c;毕竟他身上拥有庞博从妖帝坟冢带出来的道经和被誉为中州至宝的绿铜 虽然这两样物品都在叶凡的苦海中&#xff0…

【Linux】序列化与反序列化

目录 前言 什么是应用层&#xff1f; 再谈"协议" 什么是序列化和反序列化 网络版计算器 整体流程实现 Sock.hpp的实现 TcpServer.hpp的实现 Protocol.hpp的实现 CalServer.cc的编写 CalClient.cc的编写 整体代码 前言 本章是属于TCP/UDP四层模型中的第一层…

VSCode连接服务器

Pycharm连接服务器参考我的另一篇文章Pycharm远程连接服务器_pycharm进入服务器虚拟环境终端_Jumbo星的博客-CSDN博客 本质上Pycharm和VSCode都只是IDE&#xff0c;没有什么好坏之分。但是因为Pycharm连接服务器&#xff08;准确来说是部署&#xff09;需要买professional。而…

Shell 脚本入门

目录 一、Shell是什么 1.1 我们为什么要学习Shell和使用Shell&#xff1f; 1.2 Shell的分类有哪些&#xff1f; 二、Shell脚本入门知识 2.1 Shell文件命名规范 2.2 Shell解析器 2.3 用Shell 编写hello World 三、Shell的四种变量类型 3.1 系统预定义变量 3.2 自定义变…