java数据结构之排序

前言:

排序在我们日常生活中随处可见,这里将介绍java数据结构里面常见的几种排序。

ps:

swap函数的实现:

public void swap(int[] arr, int i, int j) {int tmp = arr[i];arr[i] = arr[j];arr[j] = tmp;
}

1.直接插入排序

(1)分析:

(2)代码实现:

时间复杂度:最好情况:O(N)              最坏情况:O(N^{2})

空间复杂度:O(1)

稳定性:稳定

public void insertSort(int[] arr) {for(int i = 1; i < arr.length; i++) {int j = i - 1;int tmp = arr[i];while (j >= 0) {if(arr[j] > tmp) {swap(arr,j,j+1);}else {break;}j--;}}
}

2.希尔排序

希尔排序的初衷就是想要将数组的元素变得逐渐有序,在最后一次排序时能够加快速度。希尔排序内部也是采用直接插入排序。

(1)分析:

(2)代码实现:

时间复杂度:O(N^{_{1.3}})~O(N^{1.5}) ( 经研究表明 ) 

空间复杂度:O(1)

稳定性:不稳定

private void shell(int[] arr, int gap) {for(int i = gap; i < arr.length; i++) {int j = i - gap;int tmp = arr[i];while (j >= 0) {if(arr[j] > tmp) {swap(arr,j,j+gap);}else {break;}j--;}}
}
public void shellSort(int[] arr) {int gap = arr.length;while(gap > 1) {gap = gap / 2;shell(arr,gap);}
}

3.选择排序

(1)分析:

(2)代码实现:

时间复杂度:O(N^{2})

空间复杂度:O(1)

稳定性:不稳定

public void selectSort(int[] arr) {for (int i = 0; i < arr.length; i++) {int minIndex = i;for (int j = i + 1; j < arr.length; j++) {if(arr[j] < arr[minIndex]) {minIndex = j;}}swap(arr,i,minIndex);}
}

(3)优化:

public void selectSortFastly(int[] arr) {int right = arr.length - 1;int left = 0;int minIndex = 0;int maxIndex = 0;while(left < right) {minIndex = left;maxIndex = left;for (int i = left + 1; i <= right; i++) {if(arr[i] < arr[minIndex]) {minIndex = i;}if(arr[i] > arr[maxIndex]) {maxIndex = i;}}//有一种情况需要注意,就是最大值就是第一个元素,//eg:91,2,5,3,6,4,7,9,按照下面方法会出错,在第一次交换中把最大值交换走了//swap(arr,left,minIndex);//swap(arr,right,maxIndex);swap(arr,right,maxIndex);swap(arr,left,minIndex);left++;right--;}
}

 4.堆排序

(1)分析:

(2)代码实现:

时间复杂度:O(N\log_{2}N)

空间复杂度:O(1)

稳定性:不稳定 

private void softDown(int[] arr,int parent,int size) {int child = parent * 2 + 1;while(child < size) {if(child + 1 < size && arr[child] < arr[child + 1]) {child++;}swap(arr,child,parent);parent = child;child = parent * 2 + 1;}
}
//创建大根堆
private void createMaxHeap(int[] arr) {for(int parent = (arr.length - 1 - 1) / 2; parent >= 0; parent--) {softDown(arr,parent,arr.length);}
}public void heapSort(int[] arr) {createMaxHeap(arr);int size = arr.length;while(size > 0) {swap(arr,0,size - 1);softDown(arr,0,size - 1);size--;}
}

5.冒泡排序

(1)分析:

(2)代码实现:

时间复杂度:O(N^{2})

空间复杂度:O(1)

稳定性:稳定

public void bubbleSort(int[] arr) {for (int i = 0; i < arr.length; i++) {for (int j = 0; j < arr.length - 1 - i; j++) {//从小到大排if(arr[j] > arr[j + 1]) {swap(arr,j,j+1);}}}
}

(3)优化:

给定一个标志位,如果进行一次排序过后没有进行过交换就说明整个数组已经有序,直接跳出循环。

public void bubbleSort(int[] arr) {for (int i = 0; i < arr.length; i++) {boolean flag = false;for (int j = 0; j < arr.length - 1 - i; j++) {//从小到大排if(arr[j] > arr[j + 1]) {swap(arr,j,j+1);flag = true;}}if(!flag) {//代表此时已经有序break;}}}

6.快速排序

(1)分析:


 (2)代码实现:(递归实现)

时间复杂度:O(N\log_{2}N)  但是需要注意的是当一个数组是逆序的时候,此时来进行快排,会让时间复杂度达到O(N^{2})(单只树)

空间复杂度:O(\log_{2}N)

稳定性:不稳定

//挖坑法
private int partition(int[] arr,int left, int right) {int tmp = arr[left];while(left < right) {while(left < right && arr[right] >= tmp) {right--;}arr[left] = arr[right];while(left < right && arr[left] <= tmp) {left++;}arr[right] = arr[left];}arr[left] = tmp;return left;
}
private void quick(int[] arr,int left, int right) {if(left >= right) {return;}int midIndex = partition(arr,left,right);quick(arr,left,midIndex - 1);quick(arr,midIndex + 1,right);}
public void quickSort(int[] arr) {quick(arr,0,arr.length - 1);
}

(3)优化:

我们可以使用三数取中法(在第一次partition后minIndex左右两端的元素个数是差不多相同的),和当在递归中数组的长度小于某个范围时直接用插入排序来两种方法来优化快排。

//挖坑法
private int partition(int[] arr,int left, int right) {int tmp = arr[left];while(left < right) {while(left < right && arr[right] >= tmp) {right--;}arr[left] = arr[right];while(left < right && arr[left] <= tmp) {left++;}arr[right] = arr[left];}arr[left] = tmp;return left;
}
private void quick(int[] arr,int left, int right) {if(left >= right) {return;}//优化2:if(right - left + 1 < 10) {//直接用插入排序insertSort(arr,left,right);}//优化1:三数取中法int mid = findMid(arr,left,right);swap(arr,mid,left);int midIndex = partition(arr,left,right);quick(arr,left,midIndex - 1);quick(arr,midIndex + 1,right);}
//直接插入排序
private void insertSort(int[] arr, int left, int right) {for(int i = left + 1; i <= right; i++) {int j = i - 1;int tmp = arr[i];while(j >= 0) {if(arr[j] > tmp) {swap(arr,j+1,j);}else {break;}j--;}}
}
//三数取中法
private int findMid(int[] arr, int left,int right) {int midIndex = (right - left)/2;if(arr[left] > arr[right]) {if(arr[midIndex] < arr[right]) {return right;}else if(arr[midIndex] > arr[left]) {return left;}else {return midIndex;}}else {if(arr[midIndex] < arr[left]) {return left;}else if(arr[midIndex] > arr[right]) {return right;}else {return midIndex;}}
}
public void quickSort(int[] arr) {quick(arr,0,arr.length - 1);
}

(4)非递归实现:

在这里我们会用到栈,来实现:

代码实现:

public void quickSortNor(int[] arr) {Stack<Integer> stack = new Stack<>();int left = 0;int right = arr.length-1;int midIndex = partition(arr,left,right);stack.push(midIndex+1);stack.push(right);stack.push(left);stack.push(midIndex-1);while(!stack.empty()) {right = stack.pop();left = stack.pop();if(left >= right) {continue;}midIndex = partition(arr,left,right);stack.push(midIndex+1);stack.push(right);stack.push(left);stack.push(midIndex-1);}
}

7.归并排序

(1)分析:

(2)代码实现:(递归实现)

时间复杂度:O(N\log_{2}N) 

空间复杂度:O(N)   

稳定性:稳定 

public void mergeSort(int[] arr) {mergesort(arr,0, arr.length-1);
}
private void mergesort(int[] arr, int left, int right) {if(left >= right) {return;}int mid = (left + right - 1) / 2;mergesort(arr,left,mid);mergesort(arr,mid+1,right);//合并merge(arr,left,mid,right);
}
private void merge(int[] arr, int left, int mid, int right) {int s1 = left;int e1 = mid;int s2 = mid + 1;int e2 = right;int size = right - left + 1;int[] ret = new int[size];int k = 0;while(s1 <= e1 && s2 <= e2) {if(arr[s1] < arr[s2]) {ret[k++] = arr[s1++];}else {ret[k++] = arr[s2++];}}while(s1 <= e1) {ret[k++] = arr[s1++];}while(s2 <= e2) {ret[k++] = arr[s2++];}//注意:如果在进行4个元素进行合并的时候,前四个元素合并完成,//但是如果i不加left,就会覆盖刚才放的元素。for(int i = 0; i < ret.length; i++) {arr[i+left] = ret[i];}
}

(3)非递归实现:

public void mergeSortNor(int[] arr) {int gap = 1;while(gap < arr.length) {for(int i = 0; i < arr.length; i = i + gap * 2) {int left = i;int mid = left + gap - 1;int right = mid + gap;if(mid >= arr.length) {mid = arr.length - 1;}if(right >= arr.length) {right = arr.length - 1;}merge(arr,left,mid,right);}gap *= 2;}
}

8.计数排序(不需要比较的排序)

计数排序适用排序一定范围内的数据。

(1)分析:

(2)代码实现:

public void countSort(int[] arr) {int max = arr[0];int min = arr[0];for(int i = 0; i < arr.length; i++) {if(arr[i] < min) {min = arr[i];}if(arr[i] > max) {max = arr[i];}}int size = max - min + 1;int[] count = new int[size];for(int j = 0; j < arr.length; j++) {count[arr[j] - min]++;}int k = 0;for(int i = 0; i < size; i++) {while(count[i] != 0) {arr[k++] = i + min;count[i]--;}}
}

 

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

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

相关文章

类与对象(补充)

初始化列表 1. 之前我们实现构造函数时&#xff0c;初始化成员变量主要使用函数体内赋值&#xff0c;构造函数初始化还有一种方式&#xff0c;就是初始化列表&#xff0c;初始化列表的使用方式是以一个冒号开始&#xff0c;接着是一个以逗号分隔的数据成员列表&#xff0c;每个…

【OpenCV C++20 学习笔记】序列化——XML和YAML文件处理

序列化——XML和YAML文件处理 序列化和反序列化代码实现XML/YAML文件的打开和关闭写入或读取文本和数字写入或读取OpenCV数据写入或读取数组以及map读取和写入自定义数据类型 输出结果 序列化和反序列化 如果希望永久保存某些对象&#xff0c;而不是每次运行程序的时候重新创建…

经典文献阅读之--LIV-GaussMap(实时3D辐射场地图渲染的LiDAR惯性视觉融合算法)

Tip: 如果你在进行深度学习、自动驾驶、模型推理、微调或AI绘画出图等任务&#xff0c;并且需要GPU资源&#xff0c;可以考虑使用UCloud云计算旗下的Compshare的GPU算力云平台。他们提供高性价比的4090 GPU&#xff0c;按时收费每卡2.6元&#xff0c;月卡只需要1.7元每小时&…

如何优化网站以提升UX设计质量

什么叫 UX 设计&#xff1f;UX 设计&#xff0c;即用户体验设计&#xff0c;是指为提升用户体验而进行的产品设计。 UX 在设计中&#xff0c;设计师通过调查和研究用户来掌握用户的需求和喜好&#xff0c;并利用这些信息来设计产品。设计师还会测试产品&#xff0c;以确保它们能…

学习笔记之Java篇(0725)

p this 普通方法中&#xff0c;this总是指向调用该方法的对象。 构造方法中&#xff0c;this总是指向正要初始化的对象。 this&#xff08;&#xff09;调用必须重载的构造方法&#xff0c;避免相同地址初始化代码&#xff0c;但只能在构造方法中用&#xff0c;比企鹅必须位…

不让录制的屏幕如何绕开?轻松突破录屏限制:三招搞定App录屏难题

在数字时代&#xff0c;屏幕录制已成为分享知识和记录重要信息的必备技能。然而&#xff0c;有些应用程序出于版权保护或其他原因&#xff0c;限制了屏幕录制功能。这是否意味着我们束手无策呢&#xff1f;当然不是&#xff01;本文将为您揭秘三种简单易行的方法&#xff0c;让…

html+css前端作业 王者荣耀官网1个页面(带报告)

htmlcss前端作业 王者荣耀官网1个页面&#xff08;带报告&#xff09; 下载地址 https://download.csdn.net/download/qq_42431718/89575045 目录1 目录2 项目视频 王者荣耀首页1个页面&#xff08;无js&#xff09; 页面1

【QT】SARibbon编译安装开启frameless(QWindowkit)

1.cmake开启frameless 2.检查cmakecache 3.下载编译qwindowkit 拉取saribbon时请 git clone https://github.com/czyt1988/SARibbon.git --recursive使用--recursive可以拉取第三方库 手动下载&#xff1a;https://github.com/stdware/qwindowkit 4.cmake构建 和 visual stu…

2024-07-24 Linux C語言使用inotify进行文件变化检测

一、在Linux中&#xff0c;用C语言检测文件内容变化的方法有几种&#xff0c;最常用的包括以下几种&#xff1a; 轮询&#xff08;Polling&#xff09;&#xff1a;周期性地读取文件并检查内容是否变化。inotify&#xff1a;使用Linux内核提供的inotify接口&#xff0c;这是一…

Java项目中整合多个pdf合并为一个pdf

一、Java项目中整合多个pdf合并为一个pdf gitee笔记路径&#xff1a;https://gitee.com/happy_sad/drools一、依赖导入 <dependency><groupId>com.itextpdf</groupId><artifactId>itextpdf</artifactId><version>5.5.6</version> …

Centos8 yum 更换源以及安装内核头文件

文章目录 一、简介二、yum 更换源三、安装内核头文件 一、简介 CentOS 是一个开源项目&#xff0c;发布了两个不同的 Linux 发行版——CentOS Stream 和 CentOS Linux 。 CentOS Stream 是即将发布的红帽企业 Linux 产品的上游开发平台。 CentOS 项目将于 2024 年 6 月 30 日…

嵌入式C++、ROS 、OpenCV、SLAM 算法和路径规划算法:自主导航的移动机器人流程设计(代码示例)

在当今科技迅速发展的背景下&#xff0c;嵌入式自主移动机器人以其广泛的应用前景和技术挑战吸引了越来越多的研究者和开发者。本文将详细介绍一个嵌入式自主移动机器人项目&#xff0c;涵盖其硬件与软件系统设计、代码实现及项目总结&#xff0c;并提供相关参考文献。 项目概…

基于区块链技术的高校教育资源共享的研究

&#xff08;一&#xff09;项目背景 时代变迁下的高教管理革新需求 当前&#xff0c;我国高等教育体系深受行政化管理模式影响&#xff0c;其在指引办学方向、资源优化配置及院校稳定上功不可没。然而&#xff0c;随着社会主义市场经济体系的深化发展&#xff0c;该模式逐渐显…

电脑录屏直播怎么录?3款软件推荐,达人必备

电脑录屏直播成为了一种新型、有趣且高效的传播方式。想象一下&#xff0c;当您喜欢的游戏博主进行精彩有趣的游戏直播&#xff0c;而您却因为没时间将要错过这场精彩绝伦的直播。这时&#xff0c;一款好用的录屏软件是您的必需品&#xff0c;电脑录屏能让您不再错过屏幕上的精…

学习记录701@org.hibernate.MappingException: No Dialect mapping for JDBC

使用spring data jpa 时报错&#xff1a;javax.persistence.PersistenceException: org.hibernate.MappingException: No Dialect mapping for JDBC type: 0。 但是在数据库中sql是可以执行的。 我是用的是原生查询&#xff1a; Query query entityManager.createNativeQuer…

第一百八十一节 Java IO教程 - Java文件树

Java IO教程 - Java文件树 FileVisitor API可以递归地处理文件树中的所有文件和目录。 当我们要对文件树中的所有或某些文件或目录执行某些操作时&#xff0c;FileVisitor API非常有用。 SimpleFileVisitor类是FileVisitor接口的基本实现。 当访问文件/目录时&#xff0c;Si…

React Native在移动端落地实践

在移动互联网产品迅猛发展的今天&#xff0c;技术的不断创新使得企业越来越注重降低成本、提升效率。为了在有限的开发资源下迅速推出高质量、用户体验好的产品&#xff0c;以实现公司发展&#xff0c;业界催生了许多移动端跨平台解决方案。这些方案不仅简化了开发流程&#xf…

zookeeper开启SASL权限认证

目录 一、SASL介绍 二、使用 SASL 进行身份验证 2.1 服务器到服务器的身份验证 2.2 客户端到服务器身份验证 三、验证功能 一、SASL介绍 默认情况下&#xff0c;ZooKeeper 不使用任何形式的身份验证并允许匿名连接。但是&#xff0c;它支持 Java 身份验证与授权服务(JAAS)…

景联文科技入选艾瑞咨询《2024年中国AI基础数据服务产业图谱》

2024年7月&#xff0c;国内领先的数据服务提供商景联文科技&#xff0c;成功入选艾瑞咨询发布的《2024年中国AI基础数据服务产业图谱》&#xff0c;这一荣誉不仅是对景联文科技在AI数据服务领域卓越成就的认可&#xff0c;也是对公司在未来发展中持续引领行业创新的高度期待。 …

FastAPI(七十七)实战开发《在线课程学习系统》接口开发-- 课程编辑和查看评论

源码见&#xff1a;"fastapi_study_road-learning_system_online_courses: fastapi框架实战之--在线课程学习系统" 课程编辑 先来看下课程编辑 1.判断是否登录 2.判断课程是否存在 3.是否有权限&#xff08;只有自己可以修改自己的课程&#xff09; 4.名称是否重复…