【前端算法】用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的区别
相关文章
|
4月前
|
前端开发
前端学习笔记202305学习笔记第二十八天-数组结构之快速排序4
前端学习笔记202305学习笔记第二十八天-数组结构之快速排序4
19 0
|
4月前
|
前端开发
前端学习笔记202305学习笔记第二十八天-数组结构之快速排序5
前端学习笔记202305学习笔记第二十八天-数组结构之快速排序5
18 0
|
4月前
|
前端开发
前端学习笔记202305学习笔记第二十八天-数组结构之快速排序1
前端学习笔记202305学习笔记第二十八天-数组结构之快速排序1
17 0
|
4月前
|
前端开发
前端学习笔记202305学习笔记第二十八天-数组结构之快速排序2
前端学习笔记202305学习笔记第二十八天-数组结构之快速排序2
19 0
|
10月前
|
前端开发 JavaScript
#yyds干货盘点# 前端歌谣的刷题之路-第一百六十四题-快速排序
#yyds干货盘点# 前端歌谣的刷题之路-第一百六十四题-快速排序
29 0
#yyds干货盘点# 前端歌谣的刷题之路-第一百六十四题-快速排序
|
11月前
|
JavaScript 前端开发 IDE
如何从一台新电脑配置前端常用工具实现快速开发
如何从一台新电脑配置前端常用工具实现快速开发
如何从一台新电脑配置前端常用工具实现快速开发
|
11月前
|
前端开发
前端实现导出word(docxtemplater、pizzip、jszip-utils、file-saver )
docxtemplater、pizzip、jszip-utils、file-saver 前端实现导出word
683 0
前端实现导出word(docxtemplater、pizzip、jszip-utils、file-saver )
|
11月前
|
存储 JSON JavaScript
|
11月前
|
JavaScript 前端开发
vue 前端实现随机背景色
vue 前端实现随机背景色
vue 前端实现随机背景色
|
11月前
|
前端开发 JavaScript
前端:JS实现双击table单元格变为可编辑状态
前端:JS实现双击table单元格变为可编辑状态
321 0
相关产品
云迁移中心
推荐文章
更多