十大排序 —— 冒泡排序

我们今天来讲一个大家熟悉的老朋友——冒泡排序

什么是冒泡排序

冒泡排序(Bubble Sort)是一种简单的排序算法,因其工作原理像水底下的气泡逐渐上升至水面而得名。它重复地遍历要排序的数列,比较每对相邻元素的值,如果它们的顺序错误(例如在升序排序中前者大于后者),就交换它们的位置。遍历过程会重复进行,直到整个数列变成有序状态,也就是说在某一次遍历中没有发生任何交换,这表明数列已经是有序的。

基本步骤

  1. 比较相邻元素:从数列的第一个元素开始,对每一对相邻元素做比较。
  2. 交换位置:如果第一个元素大于第二个元素(升序排序的情况下),就交换它们的位置。否则,不做交换,继续比较下一对元素。
  3. 重复上述过程:每次遍历都将未排序部分的最大值(升序时)移至末尾。
  4. 终止条件:当整个数列遍历完一遍后,如果没有发生过交换,说明数列已经是有序的,排序结束。

特点

  • 稳定性:冒泡排序是稳定的排序算法,即相等的元素在排序前后相对位置不会改变。
  • 时间复杂度:最好情况下(输入数组已经是排序好的)为O(n),但最坏和平均情况下的时间复杂度均为O(n^2),其中n是数列的长度。
  • 空间复杂度:O(1),因为排序是原地进行的,不需要额外的存储空间。
  • 效率:由于其较高的时间复杂度,冒泡排序在大规模数据集上效率较低,通常用于教学目的或小规模数据排序。

举个例子

#include <iostream>

// 冒泡排序函数
// 参数: a - 指向待排序数组的指针, size - 数组的大小
void Bubble_Sort(int *a, int size) {
    // 外层循环控制遍历的轮数,每轮将剩余未排序部分的最大元素放到末尾
    for(int i = 0; i < size; i++) {
        // 内层循环负责每一轮的具体比较和交换操作
        // 注意:由于每轮外循环后,最大的元素已经被排到了末尾,所以下一轮可以减少一次比较
        for(int j = 0; j < size - i - 1; j++) {
            // 如果当前元素大于下一个元素(这里应是降序排序,若需升序则应改为a[j] > a[j + 1])
            if(a[j] < a[j + 1]) {
                // 交换元素位置
                int temp = a[j+1]; // 临时保存较大值
                a[j+1] = a[j];     // 将较小值移到后面
                a[j] = temp;       // 将较大值移到前面
            }
        }
    }
}

// 打印数组函数
// 参数: a - 指向数组的指针, size - 数组的大小
void Print(int *a, int size) {
    // 遍历数组并打印每个元素
    for(int i = 0; i < size; i++) {
        std::cout << a[i] << " ";
    }
    // 每行打印结束后换行
    std::cout << std::endl;
}

// 程序主入口
int main() {
    // 定义并初始化一个整型数组
    int arr[] = {64, 34, 25, 12, 22, 11, 90};
    // 计算数组的元素数量
    int n = sizeof(arr)/sizeof(arr[0]);

    // 打印原始数组
    Print(arr, n);

    // 调用冒泡排序函数对数组进行排序
    Bubble_Sort(arr, n);

    // 打印排序后的数组
    Print(arr, n);

    return 0; // 程序正常结束
}

在这里插入图片描述
我们画个图:
在这里插入图片描述
当 i = 0时,就是64和面的数字比较:
在这里插入图片描述
发现90比自己大,64和90交换:
在这里插入图片描述
此时下标j到了6,一轮结束的标志就是 j + 1 < size:
在这里插入图片描述
此时开始第二轮,因为我们第一个数比较过了,我们可以从34之后的数开始,比较5个数,之后再来一轮,比较4个数…这就是内循环的写法的原因:
在这里插入图片描述

优化

  • 提前终止:设置一个标志位,用于记录在一次完整的遍历中是否发生了交换,如果没有交换,说明数组已经是有序的,可提前结束排序。
  • 记录最后一次交换的位置:下次遍历只需到该位置,因为之后的元素已经是有序的。

我们这里用提前终止:

#include <iostream>

// 冒泡排序函数,包含提前终止的优化
// 参数: a - 指向待排序数组的指针, size - 数组的大小
void Bubble_Sort(int *a, int size) {
    // 外层循环控制遍历的轮数
    for(int i = 0; i < size; i++) {
        bool swapped = false; // 添加一个标记,用于检查本轮循环中是否有交换发生

        // 内层循环负责具体比较和交换操作
        // 注意:每轮外循环后,最大的元素已经被排到了末尾,所以下一轮可以减少一次比较
        for(int j = 0; j < size - i - 1; j++) {
            // 检查当前元素是否小于下一个元素(这里实现的是降序排序,升序应为a[j] > a[j + 1])
            if(a[j] < a[j + 1]) {
                // 交换元素
                int temp = a[j+1]; // 临时保存较大值
                a[j+1] = a[j];     // 将较小值移到后面
                a[j] = temp;       // 将较大值移到前面
                swapped = true;    // 标记有交换发生
            }
        }

        // 如果在某次遍历中没有发生任何交换,说明数组已经是有序的,可以提前结束排序
        if(!swapped) break;
    }
}

// 打印数组的函数
// 参数: a - 指向数组的指针, size - 数组的元素数量
void Print(int *a, int size) {
    // 遍历数组并打印每个元素
    for(int i = 0; i < size; i++) {
        std::cout << a[i] << " ";
    }
    // 每行打印完毕后换行
    std::cout << std::endl;
}

// 主函数
int main() {
    // 初始化一个整型数组
    int arr[] = {64, 34, 25, 12, 22, 11, 90};
    // 计算数组长度
    int n = sizeof(arr)/sizeof(arr[0]);

    // 打印原始数组
    Print(arr, n);

    // 对数组进行冒泡排序
    Bubble_Sort(arr, n);

    // 打印排序后的数组
    Print(arr, n);

    return 0; // 程序正常退出
}

冒泡的各项性能

冒泡排序(Bubble Sort)作为一种基础的排序算法,其主要性能指标包括时间复杂度、空间复杂度和稳定性。下面是这些性能指标的详细解释:

时间复杂度

  1. 最好情况(最优时间复杂度):当输入数组已经是排序好的情况下,冒泡排序只需进行一轮遍历就可以判断出数组已经有序,此时的时间复杂度为 O(n),其中n是数组的长度。
  1. 最坏情况(最劣时间复杂度):当输入数组是逆序的,即每个元素都位于其正确位置的对面,冒泡排序需要进行n-1轮遍历,每轮遍历中都需要进行n-i次比较(因为每轮都会把一个元素放到最终位置,所以比较次数递减),总比较次数接近n(n-1)/2,因此最坏情况下的时间复杂度为 O(n^2)
  1. 平均时间复杂度:考虑到各种随机排列的可能性,冒泡排序的平均时间复杂度也是 O(n^2)

空间复杂度

冒泡排序是一种原地排序算法,它不需要额外的存储空间来存放数据,除了几个用于交换的临时变量。因此,它的空间复杂度为 O(1)

稳定性

冒泡排序是稳定的排序算法。这意味着相等的元素在排序前后相对位置不会改变。这是因为冒泡排序在交换元素时只会交换不满足排序条件的相邻元素,如果元素相等则不会进行交换,从而保持了稳定性。

总结

尽管冒泡排序由于其简单易懂而常用于教学,但由于其较高的时间复杂度,在处理大量数据时效率低下,通常不推荐在实际应用中使用,特别是在数据量大的场景下。在需要高效排序的场合,更高效的算法如快速排序、归并排序或堆排序等是更好的选择。然而,冒泡排序在小规模数据或者几乎已排序的数据集上可能表现得还可以接受,特别是经过优化(如添加提前终止的条件)后。

  • 10
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
冒泡排序是一种简单的排序算法,它重复地遍历待排序的元素,比较相邻元素的大小,如果顺序错误就交换它们,直到没有需要交换的元素为止。下面是冒泡排序的C语言实现: ```c void bubbleSort(int arr[], int n) { for(int i = 0; i < n-1; i++) { int flag = 0; //是否冒泡 for(int j = 0; j < n-1; j++) { if(arr[j > arr[j+1]) { swap(&arr[j], &arr[j+1]); flag = 1; } } if(flag == 0) { //如果没有发生交换,说明数组已经有序,可以提前结束循环 break; } } } ``` 在这个实现中,冒泡排序函数`bubbleSort`接受一个整型数组`arr`和数组长度`n`作为参数。它使用两层循环来遍历数组,并通过比较相邻元素的大小来进行交换。内层循环每次将最大的元素冒泡到数组的末尾。外层循环控制了冒泡的轮数,每一轮都会将当前未排序的最大元素放在正确的位置上。如果在某一轮的冒泡过程中没有发生交换,说明数组已经有序,可以提前结束循环,从而提高算法的效率。 需要注意的是,上述代码中使用了一个`swap`函数来交换两个元素的值,你可以根据需要自行实现该函数。此外,为了减少冒泡排序的时间复杂度,可以在内层循环中添加一个标志位`flag`,用于标记是否发生了交换。如果某一轮的冒泡过程中没有发生交换,说明数组已经有序,可以提前结束循环。这样可以避免不必要的比较和交换操作,提高排序的效率。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值