【JavaScript 算法】快速排序:高效的排序算法

news2024/9/28 4:13:24

在这里插入图片描述

🔥 个人主页:空白诗

在这里插入图片描述

文章目录

    • 一、算法原理
    • 二、算法实现
    • 三、应用场景
    • 四、优化与扩展
    • 五、总结

在这里插入图片描述

快速排序(Quick Sort)是一种高效的排序算法,通过分治法将数组分为较小的子数组,递归地排序子数组。快速排序通常比其他 O(n log n) 算法表现更好,因为它的内部循环可以在大多数架构上被有效地实现。本文将详细介绍快速排序算法的原理、实现及其应用。


一、算法原理

快速排序通过以下步骤实现:

  1. 选择基准:从数组中选择一个元素作为基准(pivot)。
  2. 划分数组:将数组分为两部分,一部分的元素都小于基准,另一部分的元素都大于基准。
  3. 递归排序:对划分后的两部分分别进行快速排序。
  4. 合并结果:将排序好的两部分合并,得到最终的排序结果。

在这里插入图片描述


二、算法实现

以下是快速排序的JavaScript实现:

/**
 * 快速排序算法
 * @param {number[]} arr - 需要排序的数组
 * @return {number[]} - 排序后的数组
 */
function quickSort(arr) {
  if (arr.length <= 1) {
    return arr; // 基础情况:数组为空或只有一个元素
  }
  
  const pivot = arr[Math.floor(arr.length / 2)]; // 选择基准元素
  const left = [];
  const right = [];
  
  // 将数组分为小于基准和大于基准的两部分
  for (let i = 0; i < arr.length; i++) {
    if (i === Math.floor(arr.length / 2)) continue; // 跳过基准元素
    if (arr[i] < pivot) {
      left.push(arr[i]);
    } else {
      right.push(arr[i]);
    }
  }
  
  // 递归排序并合并结果
  return quickSort(left).concat([pivot], quickSort(right));
}

// 示例
const arr = [3, 6, 8, 10, 1, 2, 1];
const sortedArr = quickSort(arr);
console.log(sortedArr); // 输出: [1, 1, 2, 3, 6, 8, 10]

三、应用场景

  1. 大规模数据排序:快速排序在处理大规模数据时表现优秀。
  2. 需要高效排序的场景:快速排序通常比其他排序算法更快,适用于需要高效排序的场景。
  3. 多种数据类型排序:快速排序可以排序各种数据类型,如数字、字符串等。

四、优化与扩展

  1. 选择基准优化:选择基准的方式可以影响排序效率。常见的优化方法包括三数取中法和随机选择法。
/**
 * 三数取中法选择基准
 * 该方法通过选择数组中的三个元素(第一个元素、中间元素和最后一个元素),
 * 并将它们进行比较,选择其中的中位数作为基准索引,以减少快速排序的最坏情况发生。
 * @param {number[]} arr - 数组
 * @return {number} - 基准索引
 */
function medianOfThree(arr) {
  const mid = Math.floor(arr.length / 2); // 计算中间索引
  const a = arr[0]; // 数组第一个元素
  const b = arr[mid]; // 数组中间元素
  const c = arr[arr.length - 1]; // 数组最后一个元素
  
  // 比较三个元素,返回中位数对应的索引
  if ((a > b) !== (a > c)) return 0; // 如果 a 既不是最大也不是最小,返回 0
  if ((b > a) !== (b > c)) return mid; // 如果 b 既不是最大也不是最小,返回 mid
  return arr.length - 1; // 否则返回最后一个元素的索引
}
  1. 尾递归优化:通过尾递归优化减少栈的深度,提高效率。
/**
 * 快速排序算法(尾递归优化)
 * 该算法通过分治法将数组分为较小的子数组,递归地排序子数组。尾递归优化可以减少栈的深度,提高效率。
 * @param {number[]} arr - 需要排序的数组
 * @return {number[]} - 排序后的数组
 */
function quickSortTailRecursive(arr) {
  /**
   * 分区函数
   * 该函数选择一个基准元素,并将数组分为两部分,一部分小于基准,另一部分大于基准
   * @param {number[]} arr - 需要排序的数组
   * @param {number} left - 左边界索引
   * @param {number} right - 右边界索引
   * @return {number} - 分区索引
   */
  function partition(arr, left, right) {
    const pivot = arr[Math.floor((left + right) / 2)]; // 选择中间元素作为基准
    let i = left; // 初始化左指针
    let j = right; // 初始化右指针

    // 左右指针向中间移动,进行分区操作
    while (i <= j) {
      while (arr[i] < pivot) i++; // 左指针右移,直到找到大于等于基准的元素
      while (arr[j] > pivot) j--; // 右指针左移,直到找到小于等于基准的元素
      if (i <= j) {
        // 交换左右指针所指向的元素
        [arr[i], arr[j]] = [arr[j], arr[i]];
        i++; // 左指针右移
        j--; // 右指针左移
      }
    }
    return i; // 返回分区索引
  }

  /**
   * 排序函数
   * 递归地对数组进行排序
   * @param {number[]} arr - 需要排序的数组
   * @param {number} left - 左边界索引
   * @param {number} right - 右边界索引
   */
  function sort(arr, left, right) {
    if (left >= right) return; // 基础情况:子数组长度为0或1时,停止递归

    const index = partition(arr, left, right); // 获取分区索引
    sort(arr, left, index - 1); // 递归排序左子数组
    sort(arr, index, right); // 递归排序右子数组
  }

  sort(arr, 0, arr.length - 1); // 初始调用排序函数,排序整个数组
  return arr; // 返回排序后的数组
}

// 示例
const arr = [3, 6, 8, 10, 1, 2, 1];
const sortedArrTailRecursive = quickSortTailRecursive(arr);
console.log(sortedArrTailRecursive); // 输出: [1, 1, 2, 3, 6, 8, 10]

五、总结

快速排序是一种高效的排序算法,通过分治法将数组分为较小的子数组,递归地排序子数组。理解和掌握快速排序算法对于处理大规模数据和优化程序性能都具有重要意义。希望本文对你理解和应用快速排序有所帮助。


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

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

相关文章

近期几首小诗汇总-生活~卷

生活 为生活飘零&#xff0c;风雨都不阻 路见盲人艰&#xff0c;为她心点灯 贺中科大家长论坛成立十五周年 科学家园有喜贺 园外丑汉翘望中 曾一学子入我科 正育科二盼长大 憧憬也能入此家 与科学家论短长 园外翘首听高论 发现有隙入此坛 竟然也能注册成 入园浏览惶然立 此贴…

使用ffmpeg将一个目录下的mkv格式的视频文件转换成mp4格式

最近学剪辑&#xff0c;从BT种子下载的素材资源都是mkv格式的&#xff0c;不能直接导入到视频剪辑软件中。这种情况下需要用一些格式转换工具进行转换&#xff0c;也可以使用ffmpeg进行编辑。 ffmpeg是一个命令行工具&#xff0c;用来对本地的音频视频软件进行编辑。ffmpeg我也…

【服务器】端口映射

文章目录 1.端口映射的概念1.1 端口映射的类型1.2 端口映射的应用场景1.3 示例 2.为什么要进行端口映射呢&#xff1f;3.原理3.1【大白话】原理解释3.2 原理图 4.代码 1.端口映射的概念 端口映射&#xff08;Port Mapping&#xff09;&#xff0c;也称为端口转发&#xff08;P…

萝卜快跑:未来出行的双刃剑

欢迎来到 破晓的历程的 博客 ⛺️不负时光&#xff0c;不负己✈️ 在这个日新月异的科技时代&#xff0c;无人驾驶技术正以前所未有的速度改变着我们的出行方式。萝卜快跑&#xff0c;作为自动驾驶出租车领域的佼佼者&#xff0c;其出现无疑为城市交通注入了新的活力&#xff…

[微信小程序知识点]自定义组件-拓展-外部样式类

使用组件时&#xff0c;组件使用者可以给组件传入css类名&#xff0c;通过传入的类名修改组件的样式 。 如果需要使用外部样式类修改组件的样式&#xff0c;在Component中需要用extemalClassess定义若干个外部样式类。 具体用法如下: (1)在Components文件里创建custom06组件 (…

谷粒商城实战笔记-26-分布式组件-SpringCloud-Gateway网关核心概念原理

微服务架构中&#xff0c;API网关扮演着至关重要的角色&#xff0c;它不仅作为微服务间的通信桥梁&#xff0c;还负责安全、监控、限流等职责。 一&#xff0c;网关的发展历程 SpringCloud的网关经历了两代的迭代和更替。 第一代网关是早期的Zuul&#xff0c;由 Netflix 开发…

【密码学】数字签名

一、数字签名的基本概念 数字签名是一种用于验证电子文档完整性和身份认证的密码学技术。它通过使用公钥加密体系中的私钥对文档的一部分&#xff08;通常是文档的摘要&#xff09;进行加密&#xff0c;从而创建一个“签名”。这个签名可以附在文档上&#xff0c;或作为一个单独…

巧用 VScode 网页版 IDE 搭建个人笔记知识库!

[ 知识是人生的灯塔&#xff0c;只有不断学习&#xff0c;才能照亮前行的道路 ] 巧用 VScode 网页版 IDE 搭建个人笔记知识库! 描述&#xff1a;最近自己在腾讯云轻量云服务器中部署了一个使用在线 VScode 搭建部署的个人Markdown在线笔记&#xff0c;考虑到在线 VScode 支持终…

C++笔试真题

可变分区管理方案 最佳适应&#xff1a;空闲区按容量递增最坏适应&#xff1a;空闲区按容量递减首先适应&#xff1a;空闲区按地址递增 C的结构体中有构造函数。 Linux新建用户或组 useradd&#xff1a;命令用于建立用户账号usermod&#xff1a;修改用户账号groupadd&#…

Spire.PDF for .NET【文档操作】演示:C#/VB.NET:压缩 PDF 文档

大型 PDF 文件处理起来很麻烦&#xff0c;占用宝贵的存储空间并减慢传输和上传速度。压缩 PDF 文档是一种简单有效的方法&#xff0c;可以减少文件大小并针对各种用途进行优化。通过压缩 PDF&#xff0c;您可以更轻松地通过电子邮件或云存储平台共享它们&#xff0c;加快下载速…

用Apipost压力测试接口

用Apipost压力测试接口 1.点击自动化测试 2.选择要测试的接口 3.如果没有接口&#xff0c;就先在api调试中添加要测试的接口 4.根据自己的需求设置相应的参数&#xff0c;这里我压测10次 5.这样就可以压测接口了&#xff0c;非常nice

接口幂等性和解决方案

针对前端重复发起相同请求的解决方案&#xff1a; 注意&#xff1a; Redis保证了在相同key的情况下&#xff0c;只会保留一条数据&#xff0c;这就保证了多次请求只会消费一条数据。 并且需要注意的是&#xff0c;生成/获取token和携带token发送请求的过程需要是两个不同的过…

showdoc sqli to rce漏洞利用思考

漏洞版本 sqli <3.2.5 phar 反序列化 <3.2.4 漏洞分析 前台sqli 补丁 https://github.com/star7th/showdoc/commit/84fc28d07c5dfc894f5fbc6e8c42efd13c976fda 补丁对比发现&#xff0c;在server/Application/Api/Controller/ItemController.class.php中将$item_id变量…

Java--抽象类

1.抽象--abstract 2.不能对抽象类进行实例化&#xff0c;也就是不能new这个抽象类 3.抽象类的应用&#xff0c;就是在class前加入abstract这个单词&#xff0c;同理抽象方法也是在void前加入abstract 4.在抽象类中可以写普通方法&#xff0c;但抽象方法只能写在抽象类中 5.…

Linux账号和权限管理详解

Linux系统中安装和管理程序 太详细了 &#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的…

婚恋交友语音交友小程序APP系统开发

在数字化时代&#xff0c;婚恋交友的方式也日益多样化。传统的相亲、朋友介绍等方式已经无法满足现代人快节毒的生活需求&#xff0c;更多的人开始选择通过线上平台寻找自己的另-婚恋交友语音交友小程序APP应运而生&#xff0c;为单身男女提供了个便捷、高效的交友平台。本文将…

vue使用quill编辑器自定义附件上传方法,并根据上传附件名称生成链接

1、附件上传 需求&#xff1a; 在编辑器中上传word,pdf,excel等附件后&#xff0c;能根据上传附件的名称生成link链接&#xff0c;在展示页面能实现点击链接下载或预览附件&#xff0c;效果图如下: 实现方法&#xff1a; quill编辑器自身带有link&#xff0c;但不满足需求&…

探索【Python面向对象】编程:新时代的高级编程范式详解

目录 1. 面向对象编程概念&#xff08;OOP&#xff09; 1.1 什么是类和对象&#xff1f; 1.2 类的定义 1.3 类和对象的关系 1.4 小李的理解 2. 抽象 2.1 抽象的概念 2.2 抽象类和方法 2.3 小李的理解 3. 类和实例 3.1 类的定义和实例化 3.2 类的属性和方法 3.3 小…

[Linux][Shell][Shell逻辑控制]详细讲解

目录 1.if 判断1.if-then2.if-then-else3.elif4.case5.实际上手 2.条件测试0.事前说明1.test 命令2.[]3.双括号1.(())2.[[]] 4.实际上手 3.循环1.for2.while3.until命令4.控制循环1.break2.continue 5.处理循环的输出 1.if 判断 1.if-then 语法&#xff1a;if command thenco…

Java技术栈总结:容器集合篇

一、List 1、ArrayList &#xff08;1&#xff09;底层数据结构 底层数据结构为数组。数组是一种用连续的内存空间存储相同数据类型数据的线性数据结构。 Q&#xff1a;为什么数组索引下标从0开始&#xff1f; A&#xff1a;从0开始&#xff0c;对应寻址公式&#xff1a;a[i]…