从小到大排序

基础排序

  1. /**
  2. * max排序
  3. * @param {*} arr
  4. * 耗时:760ms
  5. */
  6. function maxSort(arr) {
  7. let result = [...arr];
  8. for(let i=0,len=result.length; i< len; i++) {
  9. let minV = Math.min(...result.slice(i))
  10. let pos = result.indexOf(minV,i)
  11. result.splice(pos, 1)
  12. result.unshift(minV)
  13. }
  14. return result.reverse()
  15. }

冒泡排序

冒泡1

image.png
4.gif

  1. /**
  2. * 置换函数
  3. * @param {源数组} arr
  4. * @param {原数组的A项} indexA
  5. * @param {原数组的B项} indexB
  6. */
  7. function swap(arr, indexA, indexB) {
  8. [arr[indexA], arr[indexB]] = [arr[indexB], arr[indexA]];
  9. }
  10. /**
  11. * 原始冒泡排序
  12. * @param {数组} arr
  13. * 耗时:377ms
  14. */
  15. function bubbleSort1(arr) {
  16. for (let i = arr.length - 1; i > 0; i--) {
  17. for (let j = 0; j < i; j++) {
  18. if (arr[j] > arr[j + 1]) {
  19. swap(arr, j, j + 1);
  20. }
  21. }
  22. }
  23. return arr;
  24. }

冒泡排序2

  1. /**
  2. * 利用索引优化后的冒泡排序
  3. * @param {数组} arr
  4. * 耗时:350ms
  5. */
  6. function bubbleSort2(arr) {
  7. let i = arr.length - 1;
  8. while (i > 0) {
  9. let pos = 0;
  10. for (let j = 0; j < i; j++) {
  11. if (arr[j] > arr[j + 1]) {
  12. pos = j;
  13. swap(arr, j, j + 1);
  14. }
  15. }
  16. i = pos;
  17. }
  18. return arr;
  19. }

冒泡排序3

  1. /**
  2. * 在每趟排序中进行正向和反向两遍冒泡 ,
  3. * 一次可以得到两个最终值(最大和最小),
  4. * 从而使外排序趟数大概减少了一半
  5. * @param {*} arr
  6. * 耗时:312ms
  7. */
  8. function bubbleSort3(arr) {
  9. let start = 0;
  10. let end = arr.length - 1;
  11. while (start < end) {
  12. let endPos = 0;
  13. let startPos = 0;
  14. for (let i = start; i < end; i++) {
  15. if (arr[i] > arr[i + 1]) {
  16. endPos = i;
  17. swap(arr, i, i + 1);
  18. }
  19. }
  20. end = endPos;
  21. for (let i = end; i > start; i--) {
  22. if (arr[i - 1] > arr[i]) {
  23. startPos = i;
  24. swap(arr, i - 1, i);
  25. }
  26. }
  27. start = startPos;
  28. }
  29. return arr;
  30. }

选择排序

7.gif

  1. selectSort() {
  2. for (let i = 0; i < this.list.length - 1; i++) {
  3. let minIndex = i;
  4. for (let j = minIndex + 1; j < this.list.length; j++) {
  5. if (this.list[minIndex] > this.list[j]) {
  6. // 将后边小的移动到前部
  7. minIndex = j;
  8. }
  9. }
  10. this.swap(minIndex, i);
  11. }
  12. return this.list;
  13. }

二分搜索+插入排序

插入排序1
11.gif

  1. insertionSort() {
  2. for (let i = 1; i < this.list.length; i++) {
  3. let temp = this.list[i];
  4. // 定义frontMove变量,不断的寻找要插入temp的位置。
  5. // 如果出现this.list[frontMove-1]不大于temp,则终止while循环,将temp放入次位置
  6. let frontMove = i;
  7. // this.list[frontMove-1]>temp已经排好序的部分,大于要插入的temp,则继续往前查找。
  8. while (this.list[frontMove - 1] > temp && frontMove > 0) {
  9. // 此时temp比已排好元素小,把已排好的元素向后移动一位,将frontMove-1的值赋值给frontMove
  10. this.list[frontMove] = this.list[frontMove - 1];
  11. frontMove--;
  12. }
  13. // 前面排好序号的元素,找到了不大于temp的位置,此时将temp放入该位置
  14. this.list[frontMove] = temp;
  15. }
  16. return this.list;
  17. }
  1. /**
  2. * 改造二分查找,查找小于value且离value最近的值的索引
  3. * @param {*} arr
  4. * @param {*} maxIndex
  5. * @param {*} value
  6. */
  7. function binarySearch1(arr, maxIndex, value) {
  8. let min = 0;
  9. let max = maxIndex;
  10. while (min <= max) {
  11. const m = Math.floor((min + max) / 2);
  12. if (arr[m] <= value) {
  13. min = m + 1;
  14. } else {
  15. max = m - 1;
  16. }
  17. }
  18. return min;
  19. }
  20. /**
  21. * 使用二分法来优化插入排序
  22. * @param {*} arr
  23. * 耗时:86ms
  24. */
  25. function insertionSort1(arr) {
  26. for (let i = 1, len = arr.length; i < len; i++) {
  27. const temp = arr[i];
  28. const insertIndex = binarySearch1(arr, i - 1, arr[i]);
  29. for (let preIndex = i - 1; preIndex >= insertIndex; preIndex--) {
  30. arr[preIndex + 1] = arr[preIndex];
  31. }
  32. arr[insertIndex] = temp;
  33. }
  34. return arr;
  35. }

希尔排序

  1. /**
  2. * 希尔排序
  3. * 核心:通过动态定义的 gap 来排序,先排序距离较远的元素,再逐渐递进
  4. * @param {*} arr
  5. * 耗时:15ms
  6. */
  7. function shellSort(arr) {
  8. const len = arr.length;
  9. let gap = Math.floor(len / 2);
  10. while (gap > 0) {
  11. // gap距离
  12. for (let i = gap; i < len; i++) {
  13. const temp = arr[i];
  14. let preIndex = i - gap;
  15. while (arr[preIndex] > temp) {
  16. arr[preIndex + gap] = arr[preIndex];
  17. preIndex -= gap;
  18. }
  19. arr[preIndex + gap] = temp;
  20. }
  21. gap = Math.floor(gap / 2);
  22. }
  23. return arr;
  24. }
  25. console.log("shellSort", shellSort(testArr))

20.gif

快速排序

  1. const quickFn = (arr) => {
  2. // 快速排序使用了递归,首先需要定义递归的终止条件
  3. if (arr.length < 2) return arr;
  4. // 获取数组中间位置的值,作为标志位
  5. let pivotIndex = Math.floor(arr.length / 2);
  6. let pivot = arr.splice(pivotIndex, 1)[0];
  7. // 定义left【加入比标志位小的】数组,right【比标志位大的】数组
  8. let left = [],right = [];
  9. for (let i = 0; i < arr.length; i++) {
  10. // 小的放入left数组
  11. if (arr[i] < pivot) {
  12. left.push(arr[i]);
  13. } else {
  14. // 大的放入right数组
  15. right.push(arr[i]);
  16. }
  17. }
  18. // 递归调用left和right数组,并且拼接上标志位元素pivot。
  19. return quickFn(left).concat([pivot], quickFn(right));
  20. };

归并排序

  1. /**
  2. * 归并排序
  3. * @param {*} arr
  4. * 耗时 30ms
  5. */
  6. function concatSort(arr) {
  7. const len = arr.length;
  8. if (len < 2) { return arr; }
  9. const mid = Math.floor(len / 2);
  10. const left = arr.slice(0, mid);
  11. const right = arr.slice(mid);
  12. return concat(concatSort(left), concatSort(right));
  13. }
  14. function concat(left, right) {
  15. const result = [];
  16. while (left.length > 0 && right.length > 0) {
  17. // 一直对比left和right数组的第一个元素,如果left的小。则添加left到result
  18. result.push(left[0] <= right[0] ? left.shift() : right.shift());
  19. }
  20. return result.concat(left, right);
  21. }
  22. console.log('concatSort', concatSort(testArr));