js 基础排序算法 之 冒泡排序, 选择排序, 插入排序,快速排序

简介: js 基础排序算法 之 冒泡排序, 选择排序, 插入排序,快速排序

排序算法


排序算法没有优劣之分,在不同的场景中,不同的排序算法执行效率不同。


1.选择排序 Selection Sort


一次选择排序,可以将某个区间的最小值排列到该区域的第一位,具体的方式是:


1.找出该区域的最小值

2.将该值与该区域第一个值交换

3.对下一个区域重复上述过程,直到排序完成


let arr = [3, 2, 4, 6, 7, 9, 3, 1, 4];
/**
 * 选择排序
 * @param asc {Boolean} 是否升序
 * @returns {Array}
 */
Array.prototype.chooseSort = function (asc = true) {
    if (this.length < 1) return this;
    // 选择排序以升序为例: 从左侧选择一个值的索引进行对比,如果值大,那么交换索引,每一圈拿一个最大的索引,然后通过索引来交换位置
    for (let i = 0, l = this.length; i < l; i++) {
        let index = 0; // 获取一个索引,然后通过索引的值来进行对比
        for (let j = 1, jl = l - i; j < jl; j++) {
            if (asc && this[j] > this[index]) {
                index = j;
            } else if (!asc && this[j] < this[index]) {
                index = j;
            }
        }
        // 进行交换位置,每一圈拿一个最大或者最小值和数组最右边的一位交换
        [this[index], this[l - i - 1]] = [this[l - i - 1], this[index]]
    }
    return this;
}
// console.log(arr.chooseSort());
console.time('选择排序用时')
console.log(arr.chooseSort());
console.log(arr.chooseSort(false));
console.timeEnd('选择排序用时')


2.冒泡排序 Bubble Sort


一次冒泡排序,可以将某个区域序列的最大值排序到该区域的最后一位,具体的方式是:


1.将第1位和第2位比较,如果前者比后者大则交换

2.将第2位和第3位比较,如果前者比后者大则交换

3.依次类推,直到比较到该区域的最后两位

4.重复上述过程,直到序列排序完成


let arr = [3, 2, 4, 6, 7, 9, 3, 1, 4];
/**
 * 冒泡排序
 * @param asc {Boolean} 是否升序
 * @returns {Array}
 */
Array.prototype.bubbleSort = function (asc = true) {
    if (this.length < 1) return this;
    // 快速排序以升序为例,每一次选择前两个进行比较,大的放在前面,小的放后面,每一圈拿一个最大值,
    for (let i = 0, l = this.length; i < l; i++) {
        for (let j = 1, jl = l - i; j < jl; j++) {
            if (asc && this[j - 1] > this[j]) {
                // 交换位置
                [this[j - 1], this[j]] = [this[j], this[j - 1]]
            } else if (!asc && this[j - 1] < this[j]) {
                [this[j - 1], this[j]] = [this[j], this[j - 1]]
            }
        }
    }
    return this;
}
console.time('冒泡排序用时')
console.log(arr.bubbleSort());
console.log(arr.bubbleSort(false));
console.timeEnd('冒泡排序用时')


3.插入排序 Insertion Sort


将序列分为两个部分,一部分是有序的,一部分是无序的,现在要做的是,就是不断的从无序的部分取出数据,加入到有序的部分,直到整个排序完成


例如:序列[5, 7, 2, 3, 6]


1.分为有序的序列和无序的序列 (5) (7 2 3 6)

2.不断的扩充有序序列 (5 7) (2 3 6)

3.不断的扩充有序序列 (2 5 7) (3 6)

4.不断的扩充有序序列 (2 3 5 7) (6)

5.不断的扩充有序序列 (2 3 5 6 7)

6.排序完成


let arr = [3, 2, 4, 6, 7, 9, 3, 1, 4];
/**
 * 插入排序
 * @param asc {Boolean} 是否升序
 * @returns {Array}
 */
Array.prototype.insertionSort = function (asc = true) {
    if (this.length < 1) return this;
    // 插值排序以升序为例:把数据分为两个部分,先从无序部分取出数据到有序那部分进行排序
    for (let i = 1, l = this.length; i < l; i++) {
        // 获取第一位的数据, 如果不满足排序条件,需要插入到对应的位置
        if (asc && this[i] < this[i - 1]) {
            let temp = this[i];
            // 把后面的这一位需要排到前面的合适位置
            for (let j = i; j >= 0; j--) {
                if (j > 0 && this[j - 1] > temp) {
                    this[j] = this[j - 1];
                } else {
                    this[j] = temp;
                    break
                }
            }
        } else if (!asc && this[i] > this[i - 1]) {
            let temp = this[i];
            for (let j = i; j >= 0; j--) {
                if (j > 0 && this[j - 1] < temp) {
                    this[j] = this[j - 1];
                } else {
                    this[j] = temp;
                    break;
                }
            }
        }
    }
    return this;
}
// console.log(arr.insertionSort(false));
console.time('插入排序用时')
console.log(arr.insertionSort());
console.log(arr.insertionSort(false));
console.timeEnd('插入排序用时')


4.快速排序 Quick Sort


选择一个数(比如序列的最后一位)作为基准数,将整个序列排序成两部分,一部分比该数小,另一部分比该数大,基准数在中间,然后对剩余的序列做同样的事情,直到排序完成


例如:序列[5, 7, 2, 3, 6, 4]


1.选择4作为基准数,排序成为:(3, 2) 4 (7, 6, 5)

2.对于3,2, 继续使用该方式排序,得到: (2, 3) 4 (7,6,5)

3.对于7,6,5,继续使用该方式排序,得到: (2, 3) 4 (5,6,7)

4.排序完成


let arr = [3, 2, 4, 6, 7, 9, 3, 1, 4];
// 快速排序
/**
 * 快速排序
 * @param asc {Boolean} 是否升序
 * @returns {Array}
 */
Array.prototype.quickSort = function (asc = true) {
    if (this.length < 1) return this;
    // 快速排序:以升序为例, 获取中间的一个基线, 分为两边,小的在左,大的在右边, 左右两边继续选基线排序
    function __quickSort(arr, start, end) {
        if (start >= end || start >= arr.length - 1) return arr;
        let low = start, high = end, key = arr[end];
        while (low < high) {
            // 升序,
            if (asc) {
                // 地位小于高位 并且 当前的值小于基线
                while (low < high && arr[low] <= key) low++;
                // 否则当前地位不动,移动并且把当前低位指向的值赋值给高位
                arr[high] = arr[low];
                // 移动高位
                while (low < high && arr[high] > key) high--;
                // 否在赋值给低位,从新移动低位指针
                arr[low] = arr[high];
            } else {
                // 降序
                while (low < high && arr[low] >= key) low++;
                arr[high] = arr[low];
                while (low < high && arr[high] < key) high--;
                arr[low] = arr[high];
            }
        }
        // low === high,当两根指针重合, 把中间的值赋值
        arr[high] = key;
        __quickSort(arr, start, low - 1);
        __quickSort(arr, low + 1, end)
    }
    __quickSort(this, 0, this.length - 1);
    return this;
}
// console.log(arr.quickSort(false));
console.time('快速排序用时')
console.log(arr.quickSort());
console.log(arr.quickSort(false));
console.timeEnd('快速排序用时')


结果如下:


[
1, 2, 3, 3, 4,
4, 6, 7, 9
]
[
9, 7, 6, 4, 4,
3, 3, 2, 1
]
冒泡排序用时: 11.315ms
[
1, 2, 3, 3, 4,
4, 6, 7, 9
]
[
9, 7, 6, 4, 4,
3, 3, 2, 1
]
选择排序用时: 0.473ms
[
1, 2, 3, 3, 4,
4, 6, 7, 9
]
[
9, 7, 6, 4, 4,
3, 3, 2, 1
]
插入排序用时: 0.359ms
[
1, 2, 3, 3, 4,
4, 6, 7, 9
]
[
9, 7, 6, 4, 4,
3, 3, 2, 1
]
快速排序用时: 0.415ms


相关文章
|
7天前
|
搜索推荐 C语言
【排序算法】快速排序升级版--三路快排详解 + 实现(c语言)
本文介绍了快速排序的升级版——三路快排。传统快速排序在处理大量相同元素时效率较低,而三路快排通过将数组分为三部分(小于、等于、大于基准值)来优化这一问题。文章详细讲解了三路快排的实现步骤,并提供了完整的代码示例。
30 4
|
22天前
|
搜索推荐
冒泡排序算法
【10月更文挑战第19天】冒泡排序是一种基础的排序算法,虽然在实际应用中可能不是最优的选择,但对于理解排序算法的基本原理和过程具有重要意义。
|
1月前
|
算法 搜索推荐 Shell
数据结构与算法学习十二:希尔排序、快速排序(递归、好理解)、归并排序(递归、难理解)
这篇文章介绍了希尔排序、快速排序和归并排序三种排序算法的基本概念、实现思路、代码实现及其测试结果。
20 1
|
1月前
|
搜索推荐 C语言
排序算法--冒泡排序
排序算法--冒泡排序
13 0
|
24天前
|
算法 安全 数据安全/隐私保护
基于game-based算法的动态频谱访问matlab仿真
本算法展示了在认知无线电网络中,通过游戏理论优化动态频谱访问,提高频谱利用率和物理层安全性。程序运行效果包括负载因子、传输功率、信噪比对用户效用和保密率的影响分析。软件版本:Matlab 2022a。完整代码包含详细中文注释和操作视频。
|
9天前
|
算法 数据挖掘 数据安全/隐私保护
基于FCM模糊聚类算法的图像分割matlab仿真
本项目展示了基于模糊C均值(FCM)算法的图像分割技术。算法运行效果良好,无水印。使用MATLAB 2022a开发,提供完整代码及中文注释,附带操作步骤视频。FCM算法通过隶属度矩阵和聚类中心矩阵实现图像分割,适用于灰度和彩色图像,广泛应用于医学影像、遥感图像等领域。
|
10天前
|
算法 调度
基于遗传模拟退火混合优化算法的车间作业最优调度matlab仿真,输出甘特图
车间作业调度问题(JSSP)通过遗传算法(GA)和模拟退火算法(SA)优化多个作业在并行工作中心上的加工顺序和时间,以最小化总完成时间和机器闲置时间。MATLAB2022a版本运行测试,展示了有效性和可行性。核心程序采用作业列表表示法,结合遗传操作和模拟退火过程,提高算法性能。
|
11天前
|
存储 算法 决策智能
基于免疫算法的TSP问题求解matlab仿真
旅行商问题(TSP)是一个经典的组合优化问题,目标是寻找经过每个城市恰好一次并返回起点的最短回路。本文介绍了一种基于免疫算法(IA)的解决方案,该算法模拟生物免疫系统的运作机制,通过克隆选择、变异和免疫记忆等步骤,有效解决了TSP问题。程序使用MATLAB 2022a版本运行,展示了良好的优化效果。
|
10天前
|
机器学习/深度学习 算法 芯片
基于GSP工具箱的NILM算法matlab仿真
基于GSP工具箱的NILM算法Matlab仿真,利用图信号处理技术解析家庭或建筑内各电器的独立功耗。GSPBox通过图的节点、边和权重矩阵表示电气系统,实现对未知数据的有效分类。系统使用MATLAB2022a版本,通过滤波或分解技术从全局能耗信号中提取子设备的功耗信息。
|
10天前
|
机器学习/深度学习 算法 5G
基于MIMO系统的SDR-AltMin混合预编码算法matlab性能仿真
基于MIMO系统的SDR-AltMin混合预编码算法通过结合半定松弛和交替最小化技术,优化大规模MIMO系统的预编码矩阵,提高信号质量。Matlab 2022a仿真结果显示,该算法能有效提升系统性能并降低计算复杂度。核心程序包括预编码和接收矩阵的设计,以及不同信噪比下的性能评估。
27 3