Java快速排序算法、三路快排(Java算法和数据结构总结笔记)[7/20]

一、什么是快速排序算法

        快速排序的基本思想是选择一个基准元素(通常选择最后一个元素)将数组分割为两部分,一部分小于基准元素,一部分大于基准元素。

        然后递归地对两部分进行排序,直到整个数组有序。这个过程通过 partition 方法实现,它使用两个指针 i 和 j 来遍历数组,将小于基准元素的元素交换到左边,大于基准元素的元素交换到右边。

        最后,将基准元素放入正确的位置,并返回该位置作为划分点。快速排序的时间复杂度为O(nlogn)。

如下:快速排序算法的实现代码:

import java.util.Arrays;/*** @Description 快速排序算法**/
public class QuickSort {public static void main(String[] args) {int[] array = {0, 8, 9, 5, 6, 1, 4, 9, 3, 6, 3, 8, 2, 1, 7};quickSort(array, 0, array.length - 1);System.out.println("Sorted array: " + Arrays.toString(array));}// 递归函数用于执行快速排序public static void quickSort(int[] array, int low, int high) {if (low < high) {// 分割数组并获取基准元素的索引int pivotIndex = partition(array, low, high);// 递归排序左侧和右侧子数组quickSort(array, low, pivotIndex - 1);quickSort(array, pivotIndex + 1, high);}}// 函数用于分割数组并返回基准元素的索引public static int partition(int[] array, int low, int high) {// 选择基准元素(在本例中选择最后一个元素)int pivot = array[high];int i = low - 1;// 遍历数组for (int j = low; j < high; j++) {if (array[j] < pivot) {// 如果元素小于基准元素,则交换位置i++;swap(array, i, j);}}// 将基准元素放入正确的位置swap(array, i + 1, high);// 返回基准元素的索引return i + 1;}// 函数用于交换数组中的两个元素public static void swap(int[] array, int i, int j) {int temp = array[i];array[i] = array[j];array[j] = temp;}
}

排序输出:

上述代码中,就是快速排序算法的一种实现,总结来说就是:

1. 选择一个基准元素(通常是数组中的最后一个元素)。 
2. 将数组分割为两部分,一部分是小于基准元素的元素,另一部分是大于基准元素的元素。 
3. 对这两部分分别递归地应用快速排序算法,直到每个子数组都有序。 
4. 将所有子数组合并起来,得到最终排序后的数组。 
 
具体步骤实现: 
 
1. 选择基准元素。 
2. 设置两个指针,一个指向数组的起始位置(通常为low),另一个指向数组的结束位置(通常为high)。 
3. 从起始位置开始,向右移动指针,直到找到一个大于或等于基准元素的元素。 
4. 从结束位置开始,向左移动指针,直到找到一个小于或等于基准元素的元素。 
5. 如果左指针的位置小于右指针的位置,则交换这两个元素。 
6. 重复步骤3到5,直到左指针的位置大于或等于右指针的位置。 
7. 将基准元素与左指针所指向的元素进行交换,将基准元素放置在正确的位置。 
8. 递归地对基准元素左侧和右侧的子数组应用快速排序算法。 
 
快速排序的关键在于划分过程,即将数组分割为两部分。通过不断地划分和排序,最终实现整个数组的排序。快速排序的平均时间复杂度为O(nlogn),是一种高效的排序算法。

二、快速排序算法和三路快排的关系

        快速排序算法和三路快速排序算法都是基于快速排序思想的排序算法,它们之间的关系是三路快速排序算法是对快速排序算法的一种改进和优化。 
 
        快速排序算法通过选择一个基准元素,将数组分割为两部分,一部分是小于基准元素的元素,另一部分是大于基准元素的元素。然后递归地对这两部分进行排序,直到整个数组有序。 
 
        而三路快速排序算法在快速排序的基础上进行了改进。它通过选择两个基准元素,将数组划分为三部分:小于第一个基准元素、等于第一个基准元素、大于第一个基准元素的元素。然后递归地对小于和大于基准元素的两部分进行排序,而不需要再对等于基准元素的部分进行排序,因为它们已经在正确的位置上。 
 
三路快速排序算法在处理包含大量重复元素的数组时表现更好,因为它能够更有效地处理重复元素,减少了不必要的比较和交换操作。相比于快速排序算法,三路快速排序算法的时间复杂度仍然是O(nlogn),但在某些情况下,它的性能更好。 
 
总的来说,三路快速排序算法是对快速排序算法的改进,通过减少重复元素的比较和交换操作,提高了排序的效率。

三、三路快排代码实现

/*** @Description 三路快排**/
public class ThreeWayQuickSort {public static void main(String[] args) {int[] array = {9, 8, 5, 9, 2, 9, 1, 2, 3, 4, 6, 8, 4, 7};quickSort(array, 0, array.length - 1);System.out.println("排序后的数组:");for (int num : array) {System.out.print(num + " ");}}// 三路快速排序算法public static void quickSort(int[] array, int low, int high) {if (low < high) {// 将数组划分为三部分,并返回等于pivot值的范围int[] range = partition(array, low, high);// 递归地对小于pivot和大于pivot的部分进行排序quickSort(array, low, range[0] - 1);quickSort(array, range[1] + 1, high);}}// 划分数组为三部分的函数public static int[] partition(int[] array, int low, int high) {int pivot = array[low]; // 选择第一个元素作为pivotint lt = low; // 初始化lt指针指向lowint gt = high; // 初始化gt指针指向highint i = low + 1; // 初始化i指针指向low的下一个位置while (i <= gt) {if (array[i] < pivot) {swap(array, i, lt);i++;lt++;} else if (array[i] > pivot) {swap(array, i, gt);gt--;} else {i++;}}// 返回等于pivot值的范围int[] range = {lt, gt};return range;}// 交换数组中两个元素的函数public static void swap(int[] array, int i, int j) {int temp = array[i];array[i] = array[j];array[j] = temp;}
}

排序输出:

三路快速排序算法是一种高效的排序算法,它通过将数组划分为三个部分来进行排序:

小于pivot的部分、等于pivot的部分和大于pivot的部分。

下面是三路快速排序算法的实现思路步骤总结: 
 
1. 选择一个pivot元素(通常选择数组的第一个元素)。 
2. 初始化三个指针:lt指向数组的起始位置,gt指向数组的结束位置,i指向数组的起始位置的下一个位置。 
3. 从i开始遍历数组,比较当前元素与pivot的大小关系: 
   - 如果当前元素小于pivot,将它与lt指针指向的元素交换,并将lt和i指针都向右移动一位。 
   - 如果当前元素大于pivot,将它与gt指针指向的元素交换,并将gt指针向左移动一位。 
   - 如果当前元素等于pivot,将i指针向右移动一位。 
4. 重复步骤3,直到i指针遍历完整个数组。 
5. 最终,数组将被划分为三个部分:小于pivot的部分、等于pivot的部分和大于pivot的部分。 
6. 递归地对小于pivot和大于pivot的部分进行三路快速排序。 
 
三路快排算法通过将相等的元素放在中间,这样就避免了重复比较和交换的过程,从而提高了排序的效率。

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

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

相关文章

私域流量搭建与运营,技巧全攻略!

2023年是比拼运营深度和服务效率的一年&#xff0c;用户对于体验的期望值将持续增长&#xff0c;企业需提供无缝的客户体验&#xff0c;以推动增长、保障收入、确保客户忠诚度。在疫情新常态下&#xff0c;企业已构建起APP、小程序等一系列线上触点矩阵&#xff0c;而各个触点之…

浅谈开口互感器在越南美的工业云系统中的应用

摘 要&#xff1a;分析低压开口式电流互感器的原理&#xff0c;结合工程实例分析开口电流互感器在低压配电系统中&#xff0c;主要是改造项目中的应用及施工细节&#xff0c;为用户快速实现智能配电提供解决方案&#xff0c;该方案具有成本低、投资少、安装接线简便等优点&…

Docker Stack部署应用详解+Tomcat项目部署详细实战

Docker Stack 部署应用 概述 单机模式下&#xff0c;可以使用 Docker Compose 来编排多个服务。Docker Swarm 只能实现对单个服务的简单部署。而Docker Stack 只需对已有的 docker-compose.yml 配置文件稍加改造就可以完成 Docker 集群环境下的多服务编排。 stack是一组共享…

MySQL -- 索引

MySQL – 索引 文章目录 MySQL -- 索引一、索引简介1.简介2.索引效率的案例 二、认识磁盘1.磁盘2.结论3.磁盘随机访问(Random Access)与连续访问(Sequential Access) 三、MySQL 与磁盘交互基本单位1.基本单位2.MySQL中的数据管理 五、索引的理解1.索引案例2.单页mysql page3.管…

库房管理软件采购申请流程代码实现解析

300rmb掏了个javavue2的小系统&#xff0c;学习代码&#xff0c;调整下申请流程。 原有的入库流程是&#xff0c;库管&#xff08;admin&#xff09;提出采购申请给采购员&#xff08;caigou&#xff09;&#xff0c;采购员采购入库时点击入库完成采购入库流程。 想弄清后端代…

figma-如何批量修改字体

一.选择字体 二.批量替换 编辑—>替换相同字体

微信小程序之自定义组件开发

1、前言 从小程序基础库版本 1.6.3 开始&#xff0c;小程序支持简洁的组件化编程。所有自定义组件相关特性都需要基础库版本 1.6.3 或更高。开发者可以将页面内的功能模块抽象成自定义组件&#xff0c;以便在不同的页面中重复使用&#xff1b;也可以将复杂的页面拆分成多个低耦…

HackTheBox-Starting Point--Tier 1---Funnel

文章目录 一 题目二 实验过程三 利用SSH隧道3.1 本地端口转发 一 题目 Tags FTP、PostgreSQL、Reconnaissance、Tunneling、Password Spraying、Port Forwarding、Anonymous/Guest Access、Clear Text Credentials译文&#xff1a;FTP、PostgreSQL、侦察、隧道技术、密码喷洒…

linux内的循环

格式 while 【 条件判断 】 do 语句体 done 上图 第一次代码&#xff0c;输入语句在外面&#xff0c;结果输入完&#xff08;非hello&#xff09;程序不断循环&#xff0c;没办法&#xff0c;ctrlc给程序终止了&#xff0c;然后把用户输入的语句放到了循环体里面…

NOIP2023模拟12联测33 A. 构造

NOIP2023模拟12联测33 A. 构造 文章目录 NOIP2023模拟12联测33 A. 构造题目大意思路code 题目大意 构造题 思路 想一种构造方法&#xff0c;使得 y y y 能够凑成尽可能多的答案 第一行 x y r y ⋯ r xyry \cdots r xyry⋯r 第二行 r y x y ⋯ x ryxy \cdots x ryxy⋯x …

C语言100~200中不能整除3的数

#define _CRT_SECURE_NO_WARNINGS 1#include <stdio.h> int main() {int n;for (n 100; n < 200; n){if (n%3 0){continue;}printf("%d\n",n);}}

【亚马逊云科技产品测评】活动征文|亚马逊云科技AWS之EC2详细测评

引言 &#xff08;授权声明&#xff1a;本篇文章授权活动官方亚马逊云科技文章转发、改写权&#xff0c;包括不限于在 Developer Centre, 知乎&#xff0c;自媒体平台&#xff0c;第三方开发者媒体等亚马逊云科技官方渠道&#xff09; 在当前的数字化时代&#xff0c;云服务已…

RISC-V处理器设计(四)—— Verilog 代码设计

一、前言 从6月底刚开始接触 risc-v 架构&#xff0c;到现在完成了一个 risc-v cpu 的设计&#xff0c;并且成功移植了 rt-thread nano 到本 cpu 上运行&#xff0c;中间经过了 4个多月的时间&#xff0c;遇到了数不清的问题&#xff0c;也想过放弃&#xff0c;但好在最后还是…

Node.js |(五)包管理工具 | 尚硅谷2023版Node.js零基础视频教程

学习视频&#xff1a;尚硅谷2023版Node.js零基础视频教程&#xff0c;nodejs新手到高手 文章目录 &#x1f4da;概念介绍&#x1f4da;npm&#x1f407;安装npm&#x1f407;基本使用&#x1f407;生产依赖与开发依赖&#x1f407;npm全局安装&#x1f407;npm安装指定包和删除…

比SAM小60倍的分割一切模型:MobileSAM

1 MobileSAM SAM就是一类处理图像分割任务的通用模型。与以往只能处理某种特定类型图片的图像分割模型不同&#xff0c;SAM可以处理所有类型的图像。 在SAM出现前&#xff0c;基本上所有的图像分割模型都是专有模型。比如&#xff0c;在医学领域&#xff0c;有专门分割核磁图…

国外住宅IP代理选择的8个方法,稳定的海外IP哪个靠谱?

一、国外住宅IP代理是什么&#xff1f; 代理服务器充当您和互联网之间的网关。它是一个中间服务器&#xff0c;将最终用户与他们浏览的网站分开。如果您使用国外代理IP&#xff0c;互联网流量将通过国外代理服务器流向您请求的地址。然后&#xff0c;请求通过同一个代理服务器…

K8s Error: ImagePullBackOff 故障排除

Error: ImagePullBackOff 故障排除 1. 起因 起因是要在一组k8s环境下做个Prometheus的测试,当时虚拟机用完直接暂停了. 启动完master和node节点后重启了这些节点. 当检查dashboard时候发现Pod处于ImagePullBackOff状态,使用命令查看详细情况 kubectl describe pods -n kuber…

二十、泛型(2)

本章概要 泛型接口泛型方法 变长参数和泛型方法一个泛型的 Supplier简化元组的使用一个 Set 工具 泛型接口 泛型也可以应用于接口。例如 生成器&#xff0c;这是一种专门负责创建对象的类。实际上&#xff0c;这是 工厂方法 设计模式的一种应用。不过&#xff0c;当使用生成…

Requests 与接口请求构造

Requests 是一个优雅而简单的 Python HTTP 库&#xff0c;其实 Python 内置了用于访问网络的资源模块&#xff0c;比如urllib&#xff0c;但是它远不如 Requests 简单优雅&#xff0c;而且缺少了许多实用功能。所以&#xff0c;更推荐掌握 Requests 接口测试实战技能&#xff0…