记录几种排序算法

十种常见排序算法可以分类两大类别:比较类排序非比较类排序。      

  

        常见的快速排序归并排序堆排序以及冒泡排序等都属于比较类排序算法。比较类排序是通过比较来决定元素间的相对次序,其时间复杂度不能突破 O(nlogn)。在冒泡排序之类的排序中,问题规模为 n,又因为需要比较 n 次,所以平均时间复杂度为 O(n²)。在归并排序快速排序之类的排序中,问题规模通过分治法消减为 logn 次,所以时间复杂度平均 O(nlogn)。

        非比较类排序不通过比较来决定元素间的相对次序,而是通过确定每个元素之前,应该有多少个元素来排序。非比较排序时间复杂度底,但由于非比较排序需要占用空间来确定唯一位置。所以对数据规模和数据分布有一定的要求。

1、冒泡排序

        冒泡排序是一种简单的排序算法。它重复地遍历要排序的序列,依次比较两个元素,如果它们的顺序错误就把它们交换过来。遍历序列的工作是重复地进行直到没有再需要交换为止,此时说明该序列已经排序完成。这个算法的名字由来是因为越小的元素会经由交换慢慢 “浮”到数列的顶端。

实现代码:

/*** 冒泡排序* @param arr* @return arr*/
public static int[] bubbleSort(int[] arr) {for (int i = 1; i < arr.length; i++) {boolean flag = true;for (int j = 0; j < arr.length - i; j++) {if (arr[j] > arr[j + 1]) {int tmp = arr[j];arr[j] = arr[j + 1];arr[j + 1] = tmp;flag = false;}}if (flag) {break;}}return arr;
}

         此处对代码做了一个小优化,加入了一个Flag变量,当原输入序列就是排序好的情况下,停止遍历,此时时间复杂度为O(n)。

算法分析:

  • 稳定性:稳定
  • 时间复杂度:最佳:O(n) , 最差:O(n2),平均:O(n2)
  • 空间复杂度:O(1)

2、选择排序

        选择排序的思想更加朴实:首先在未排序序列中找到最小(大)元素,存放到排序序列的起始位置,然后,再从剩余未排序元素中继续寻找最小(大)元素,然后放到已排序序列的末尾。以此类推,直到所有元素均排序完毕。其时间复杂度永远都是O(n2)。

实现代码:

/*** 选择排序* @param arr* @return arr*/
public static int[] selectionSort(int[] arr) {for (int i = 0; i < arr.length - 1; i++) {int minIndex = i;for (int j = i + 1; j < arr.length; j++) {if (arr[j] < arr[minIndex]) {minIndex = j;}}if (minIndex != i) {int tmp = arr[i];arr[i] = arr[minIndex];arr[minIndex] = tmp;}}return arr;
}
  • 稳定性:不稳定
  • 时间复杂度:最佳:O(n2) , 最差:O(n2),平均:O(n2)
  • 空间复杂度:O(1)

3、快速排序

        快速排序的基本思想:通过一趟排序将待排序列分隔成独立的两部分,其中一部分记录的元素均比另一部分的元素小,则可分别对这两部分子序列继续进行排序,以达到整个序列有序。

算法步骤:

  • 从序列中随机挑出一个元素,做为 “基准”(pivot);
  • 重新排列序列,将所有比基准值小的元素摆放在基准前面,所有比基准值大的摆在基准的后面(相同的数可以到任一边)。在这个操作结束之后,该基准就处于数列的中间位置。
  • 递归地把小于基准值元素的子序列和大于基准值元素的子序列进行快速排序。

实现代码:

public static void quick_sort(int nums[], int start, int end){if(start >= end) return;int left = start;int right = end;int base = nums[start];while(left < right){while(left < right && nums[right] >= base){right--;}nums[left] = nums[right];while(left < right && nums[left] <= base){left++;}nums[right] = nums[left];}//此时left = rightnums[left] = base;quick_sort(nums,start,left-1);quick_sort(nums,left+1,end);}
  • 稳定性:不稳定
  • 时间复杂度:最佳:O(nlogn) , 最差:O(nlogn),平均:O(nlogn)
  • 空间复杂度:O(logn)

4、堆排序

        堆排序(Heap Sort)是指利用这种数据结构所设计的一种排序算法。堆是一个近似完全二叉树的结构,并同时满足堆积的性质:即子结点的键值或索引总是小于(或者大于)它的父节点。堆的底层是一棵完全二叉树。而完全二叉树是一层一层按照进入的顺序排成的。按照这个特性,我们可以用数组来按照完全二叉树实现堆。

算法步骤:

  1. 构建初始堆,将待排序列构成一个大顶堆(或者小顶堆),升序大顶堆,降序小顶堆;
  2. 将堆顶元素与堆尾元素交换,并断开(从待排序列中移除)堆尾元素。
  3. 重新构建堆。
  4. 重复2~3,直到待排序列中只剩下一个元素(堆顶元素)。

代码实例:

import java.util.Arrays;/*** @description: 手撕堆排序* @author: Jay* @create: 2024-05-02 11:48**/
public class heapSort {//堆的长度大小static int heapLen;//定义 交换数组两元素 的函数private static void swap(int[] arr, int a, int b){int temp = arr[a];arr[a] = arr[b];arr[b] = temp;}//初始化构建大顶堆private static void buildHeap(int[] arr){//从最后一个非叶子节点开始,对应的索引为 arr.length / 2 - 1for(int i=arr.length/2-1; i>=0; i--){adjustHeap(arr,i);}}//调整堆private static void adjustHeap(int[] arr, int i){int left = i * 2 + 1;int right = i * 2 + 2;int largest = i;if(right < heapLen && arr[right] > arr[largest]){largest = right;}if(left < heapLen && arr[left] > arr[largest]){largest = left;}if(largest != i){swap(arr,largest,i);adjustHeap(arr,largest); //交换之后,子节点可能不满足堆的性质了,需要重新调整。}}//测试public static void main(String[] args) {int[] arr = {16,7,3,20,17,8};heapLen = arr.length;buildHeap(arr);//交换堆顶和堆尾元素,再重新调整堆。for(int i=arr.length-1; i>0; i--){swap(arr,i,0);heapLen--;adjustHeap(arr,0);}System.out.println(Arrays.toString(arr));}
}

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

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

相关文章

扩展学习|本体研究进展

文献来源&#xff1a; 王向前,张宝隆,李慧宗.本体研究综述[J].情报杂志,2016,35(06):163-170. 一、本体的定义 本体概念被引入人工智能、知识工程等领域后被赋予了新的含义。然而不同的专家学者对本体的理解不同,所给出的定义也有所差异。 人工智能领域的学者Neches(1991)等人对…

Java ~ Lock【目录】

一 Lock&#xff08;锁&#xff09; Lock&#xff08;锁&#xff09; 《Java ~ Lock【源码】》《Java ~ Lock【总结】》 未完待续…

一些水到渠成的困惑

一、纹波和噪声 什么是纹波&#xff1f;什么是噪声&#xff1f; 两者是怎么产生的&#xff1f; 有什么区别与联系&#xff1f; 在电路中怎么规避&#xff1f;什么样的电路设计&#xff1f; 1.1电源纹波到底是什么&#xff1f; 二、EMC、EMI、ESR 三、二极管、电容不一样的…

Docker Compose 部署若依前后端分离版

准备一台服务器 本次使用虚拟机&#xff0c;虚拟机系统 Ubuntu20.04&#xff0c;内存 4G&#xff0c;4核。 确保虚拟机能连接互联网。 Ubuntu20.04 安装 Docker 添加 Docker 的官方 GPG key&#xff1a; sudo apt-get update sudo apt-get install ca-certificates curl su…

Spring Boot应用部署 - Tomcat容器替换为Jetty容器

Jetty和Tomcat容器对比 Tomcat和Jetty都是一种Servlet引擎&#xff0c;他们都支持标准的servlet规范和JavaEE的规范。 Jetty更轻量级。这是相对Tomcat而言的。 Jetty更灵活。 Jetty更满足公有云的分布式环境的需求&#xff0c;而Tomcat更符合企业级环境。 Tomcat容器替换为…

初始面相对象

初始面向对象 类和对象的关系 类&#xff1a;对对象向上抽取出像的部分、公共的部分以此形成类&#xff0c;类就相当于一个模版。 对象&#xff1a;在某个模版下的具体的产物可以理解为对象&#xff0c;对象就是一个一个具体的实例&#xff0c;就相当于这个模版下具体的产品&…

RabbitMQ之生产批量发送

为什么要用生产批量发送&#xff1f; 批量发送消息&#xff0c;可以提高MQ发送性能。但是 RabbitMQ 并没有提供了批量发送消息的 API 接口,使用 spring-amqp 的 BatchingRabbitTemplate 实现批量能力。 SimpleBatchingStrategy 发送策略满足以下规则会进行发送&#xff1a; ba…

学习【Java反射】这一篇就够了

反射 1. 什么是反射2. 反射的原理3. 使用案例4. 应用场景 1. 什么是反射 Java反射是指在运行时动态地获取类的信息&#xff0c;并可以通过该信息来操作类或对象。 通过反射&#xff0c;我们可以在运行时获取类的字段、方法、构造函数等信息&#xff0c;并能够动态地创建对象、…

JAVA基础JSP之JavaBean模式

JavaBean模式 1 JavaBean简介及设计原则 【JavaBean简介】符合某种特定的规范的Java类&#xff0c;使用Javabean的好处是解决代码重复编写&#xff0c;减少代码冗余&#xff0c;功能区分明确&#xff0c;提高了代码的维护性。[S1] 1.for循环 2.jdbc中的dao模式 【Javabean…

微信小程序进阶之路:项目管理与分包加载实战指南

微信小程序进阶之路&#xff1a;项目管理与分包加载实战指南 在微信小程序的开发海洋中&#xff0c;随着项目的日益复杂&#xff0c;如何高效管理项目结构和优化加载性能成为每位开发者必修课。本文专为“小白”开发者设计&#xff0c;将深入浅出地讲解项目管理的基本概念、分…

梅大(龙)高速周边地形

最近广东高速的事故很受关注&#xff0c;我下载了这个高速的地形数据。查看了一下高速周围的地形情况。确实地形很险要&#xff0c;开车还是不要太快&#xff01;尤其南方的路基不稳&#xff01;这样险要的地形很危险&#xff01; 高速周围的地形情况 梅大&#xff08;龙&…

eNSP-动态路由(ospf协议)

一、拓扑结构搭建 二、主机配置 pc1 pc2 三、路由器配置 1.AR2配置 <Huawei>sys #进入系统视图 [Huawei]int g0/0/0 #进入接口 [Huawei-GigabitEthernet0/0/0]ip address 192.168.0.2 24 #设置ip地址 [Huawei-GigabitEthernet0/0/0]q #返回上一级 [Huawei]int g0/0/1 …

关于 Vue.js 双向数据绑定基本实现认知

写在前面 很早的一篇博客&#xff0c;整理了部分&#xff0c;蹭假期整理完博文内容涉及:双向数据绑定 实现方式简单介绍基于发布订阅、数据劫持的双向数据绑定两种不同实现(ES5/ES6) Demo&#xff0c;以及代码简单分析Object.defineProperty && Proxy API 介绍以及特性…

rknn adb shell error: closed

博主的答案&#xff1a; 【Android测试】adb shell回车后出现 error closed的解决办法-CSDN博客 第1种&#xff1a;重启电脑&#xff0c;之后把手机查到电脑上&#xff0c;启动idea 第2种&#xff1a;手机-设置-应用程序-开发-usb调试打开再关闭一次 第3种&#xff1a;重启手…

Libcity笔记:原子文件

1 介绍 Libcity中的数据以原子文件的形式存在 2 原子文件类别 对于不同的交通预测任务&#xff0c;可能用到不同的原子文件&#xff0c;同一个数据集不一定包含全部六种原子文件 网格数据需要按照先行后列的顺序遍历OD数据需要按照先起点后终点的顺序遍历 2.1 geo 存储地理…

opengauss概述-基础知识篇-备考华为高斯

目录 &#x1f9e8;考前准备: &#x1f3a1;数据库操作语言 ✨OLTP和OLAP &#x1f3af;常用函数 &#x1f9f2;字符处理函数 关于 left 和 right 特别重点的字符串函数 &#x1f9f2;数字操作函数 关于 ceil 和 floor &#x1f9f2;时间和日期处理函数 &#x1f9f…

redis保存数据的结构-redisobject结构体

在redis中&#xff0c;所有键值对的保存&#xff0c;都是机遇redisboject的一个结构体&#xff0c;如下 typedef struct redisObject {unsigned type:4; unsigned encoding:4; unsigned lru:LRU_BITS; int refcount; void *ptr; …

美团面试(一面)

前言 给位小伙伴好&#xff0c;这里呢&#xff0c;分享一下最近一次美团的面试的面经&#xff0c;自己把面试的大多数内容通过博客的形式记录了下来&#xff0c;希望对各位有所帮助哦~ 一、项目篇 1、**对于自己的点餐小程序数据库表是怎么设计的 2、对于多个人下订单的问题…

Java---类和方法的再研究

上一篇主要介绍了面向对象的思想以及内存实现&#xff0c;关于类与对象感觉写的不够好&#xff0c;因此才会有这一篇作为补充&#xff1b; 一&#xff1a;类与对象 &#xff08;1&#xff09;类 一些相同属性和行为的事物的统称&#xff0c;比较广泛、抽象&#xff0c;比如…

专项技能训练五《云计算网络技术与应用》实训7-1:安装mininet

文章目录 mininet安装1. 按6-1教程安装opendaylight控制器。2. 按6-2教程安装RYU控制器。3. 按5-1教程安装openvswitch虚拟交换机并开启服务。4. 将老师所给mininet安装包试用winSCP传送至电脑端。5. 安装net-tools。6. 安装mininet7. 安装完成后&#xff0c;使用命令建立拓扑&…