【前端算法】用JS实现快速排序

简介: 理解数组方法里面运用到的算法,splice 和 slice的区别

固定算法,固定思路

  • 找到中间位置midValue
  • 遍历数组,小于midValue放在left,否则放在right
  • 继续递归。最后concat拼接,返回

细节

  • 获取midValue的两种方式:
  • 使用splice,会修改原数组
  • 使用slice,不会修改原数组——更加推荐

代码实现

  • splice方式
function quickSort1 (arr:number[]):number[] {
  const len = arr.length
  if (len === 0) return arr
   
  const midIdx = Math.floor(len / 2)
  const midValue = arr.splice(midIdx,1)[0]

  const left:number[] = []
  const right:number[] = []

  // 由于splice改变了原数组,所以不能使用len作为循环条件
  for (let i = 0; i < arr.length; i++) {
    
    if (arr[i] < midValue) {
      // 小于中间值,放left
      left.push(arr[i])
    } else {
      // 大于或者等于,放right
      right.push(arr[i])
    }
    
  }

  return quickSort1(left).concat([midValue],quickSort1(right))
}
  • slice方式
function quickSort2 (arr:number[]):number[] {
  const len = arr.length
  if (len === 0) return arr
   
  const midIdx = Math.floor(len / 2)
  const midValue = arr.slice(midIdx,midIdx + 1)[0]

  const left:number[] = []
  const right:number[] = []

  for (let i = 0; i < len; i++) {
    if (i !== midIdx) {
      if (arr[i] < midValue) {
        // 小于中间值,放left
        left.push(arr[i])
      } else {
        // 大于或者等于,放right
        right.push(arr[i])
      }
    }
    
    
  }

  return quickSort2(left).concat([midValue],quickSort2(right))
}

功能测试

  • 测试splice方式
const arr1 = [1,6,2,4,3,7,5,8,9]

quickSort1(arr1)

结果

[1, 2, 3, 4, 5, 6, 7, 8, 9]
  • 测试slice方式
const arr1 = [1,6,2,4,3,7,5,8,9]

quickSort2(arr1)

结果

[1, 2, 3, 4, 5, 6, 7, 8, 9]

单元测试

describe('快速排序', () => {
  it('正常情况', () => {
    const arr = [1, 6, 2, 4, 3, 7, 5, 8, 9]
    const res = quickSort1(arr)
    expect(res).toEqual([1, 2, 3, 4, 5, 6, 7, 8, 9])

    const arr2 = [1, 6, 2, 4, 3, 7, 5, 8, 9]
    const res2 = quickSort2(arr2)
    expect(res2).toEqual([1, 2, 3, 4, 5, 6, 7, 8, 9])
  })

  it('有负数', () => {
    const arr = [-1, -6, 2, 4, 3, 7, 5, 8, 9]
    const res = quickSort1(arr)
    expect(res).toEqual([-6, -1, 2, 3, 4, 5, 7, 8, 9])

    const arr2 = [-1, -6, 2, 4, 3, 7, 5, 8, 9]
    const res2 = quickSort2(arr2)
    expect(res2).toEqual([-6, -1, 2, 3, 4, 5, 7, 8, 9])
  })

  it('数值一样', () => {
    const arr = [2, 2, 2, 2]
    const res = quickSort1(arr)
    expect(res).toEqual([2, 2, 2, 2])

    const arr2 = [2, 2, 2, 2]
    const res2 = quickSort2(arr2)
    expect(res2).toEqual([2, 2, 2, 2])
  })

  it('空数组', () => {
    const res = quickSort1([])
    expect(res).toEqual([])

    const res2 = quickSort2([])
    expect(res2).toEqual([])
  })
})

性能测试

const test1 = []
for (let i = 0; i < 100 * 10000; i++) {
  const n = Math.floor(Math.random() * 10000)
  test1.push(n)
}
console.time('quickSort1')
quickSort1(test1)
console.timeEnd('quickSort1')

const test2 = []
for (let i = 0; i < 100 * 10000; i++) {
  const n = Math.floor(Math.random() * 10000)
  test2.push(n)
}

console.time('quickSort2')
quickSort2(test2)
console.timeEnd('quickSort2')

打印结果

quickSort1: 713.186ms
quickSort2: 685.652ms

复杂度分析

  • 有遍历,有二分——O(n*logn) 或者O(nlogn)
  • (常规排序,嵌套循环,复杂度是O(n^2))

splice 和 slice 没有区分出来

  • 算法本身的时间复杂度就够高O(n*logn)
  • 外加,splice是逐步 二分 之后执行的,二分会快速削减数量级
  • 如果单独比较splice和slice,效果会非常明显
const test1 = []
for (let i = 0; i < 100 * 10000; i++) {
  const n = Math.floor(Math.random() * 10000)
  test1.push(n)
}

console.time('splice')
test1.splice(50 * 10000 , 1)
console.timeEnd('splice')

const test2 = []
for (let i = 0; i < 100 * 10000; i++) {
  const n = Math.floor(Math.random() * 10000)
  test2.push(n)
}

console.time('slice')
test1.slice(50 * 10000 ,50 * 10000 + 1)
console.timeEnd('slice')

打印结果

splice: 0.657ms
slice: 0.021ms

slice 本身优于splice

总结

  • 常见排序算法
  • 有二分,时间复杂度就包含O(logn)
  • 注意数组操作:splice 和 slice的区别
相关文章
|
6月前
|
存储 监控 算法
局域网监控其他电脑的设备信息管理 Node.js 跳表算法
跳表通过分层索引实现O(logn)的高效查询、插入与删除,适配局域网监控中设备动态接入、IP映射及范围筛选等需求,相比传统结构更高效稳定,适用于Node.js环境下的实时设备管理。
225 9
|
6月前
|
存储 监控 算法
电脑管控软件的进程优先级调度:Node.js 红黑树算法
红黑树凭借O(log n)高效插入、删除与查询特性,适配电脑管控软件对进程优先级动态调度的高并发需求。其自平衡机制保障系统稳定,低内存占用满足轻量化部署,显著优于传统数组或链表方案,是实现关键进程资源优先分配的理想选择。
310 1
|
6月前
|
存储 监控 JavaScript
企业上网监控系统的恶意 URL 过滤 Node.js 布隆过滤器算法
布隆过滤器以低内存、高效率特性,解决企业上网监控系统对百万级恶意URL实时检测与动态更新的难题,通过概率性判断实现毫秒级过滤,内存占用降低96%,适配大规模场景需求。
355 3
|
7月前
|
运维 监控 JavaScript
基于 Node.js 图结构的局域网设备拓扑分析算法在局域网内监控软件中的应用研究
本文探讨图结构在局域网监控系统中的应用,通过Node.js实现设备拓扑建模、路径分析与故障定位,提升网络可视化、可追溯性与运维效率,结合模拟实验验证其高效性与准确性。
400 3
|
8月前
|
存储 监控 JavaScript
基于布隆过滤器的 Node.js 算法在局域网电脑桌面监控设备快速校验中的应用研究
本文探讨了布隆过滤器在局域网电脑桌面监控中的应用,分析其高效空间利用率、快速查询性能及动态扩容优势,并设计了基于MAC地址的校验模型,提供Node.js实现代码,适用于设备准入控制与重复数据过滤场景。
293 0
|
11月前
|
监控 算法 JavaScript
基于 JavaScript 图算法的局域网网络访问控制模型构建及局域网禁止上网软件的技术实现路径研究
本文探讨局域网网络访问控制软件的技术框架,将其核心功能映射为图论模型,通过节点与边表示终端设备及访问关系。以JavaScript实现DFS算法,模拟访问权限判断,优化动态策略更新与多层级访问控制。结合流量监控数据,提升网络安全响应能力,为企业自主研发提供理论支持,推动智能化演进,助力数字化管理。
278 4
|
11月前
|
监控 算法 JavaScript
公司局域网管理视域下 Node.js 图算法的深度应用研究:拓扑结构建模与流量优化策略探析
本文探讨了图论算法在公司局域网管理中的应用,针对设备互联复杂、流量调度低效及安全监控困难等问题,提出基于图论的解决方案。通过节点与边建模局域网拓扑结构,利用DFS/BFS实现设备快速发现,Dijkstra算法优化流量路径,社区检测算法识别安全风险。结合WorkWin软件实例,展示了算法在设备管理、流量调度与安全监控中的价值,为智能化局域网管理提供了理论与实践指导。
276 3
|
11月前
|
存储 监控 算法
公司内部网络监控中的二叉搜索树算法:基于 Node.js 的实时设备状态管理
在数字化办公生态系统中,公司内部网络监控已成为企业信息安全管理体系的核心构成要素。随着局域网内终端设备数量呈指数级增长,实现设备状态的实时追踪与异常节点的快速定位,已成为亟待解决的关键技术难题。传统线性数据结构在处理动态更新的设备信息时,存在检索效率低下的固有缺陷;而树形数据结构因其天然的分层特性与高效的检索机制,逐渐成为网络监控领域的研究热点。本文以二叉搜索树(Binary Search Tree, BST)作为研究对象,系统探讨其在公司内部网络监控场景中的应用机制,并基于 Node.js 平台构建一套具备实时更新与快速查询功能的设备状态管理算法框架。
363 3
|
12月前
|
算法 搜索推荐
快速排序-数据结构与算法
快速排序(Quick Sort)是一种基于分治法的高效排序算法。其核心思想是通过选择基准(pivot),将数组划分为左右两部分,使得左侧元素均小于基准,右侧元素均大于基准,然后递归地对左右两部分进行排序。时间复杂度平均为 O(n log n),最坏情况下为 O(n²)(如数组已有序)。空间复杂度为 O(1),属于原地排序,但稳定性不佳。 实现步骤包括编写 `partition` 核心逻辑、递归调用的 `quickSort` 和辅助函数 `swap`。优化方法有随机化基准和三数取中法,以减少最坏情况的发生。
728 13
|
12月前
|
JavaScript 前端开发 API

热门文章

最新文章

下一篇
开通oss服务