首页 > 其他分享 >【C语言】全面解析冒泡排序

【C语言】全面解析冒泡排序

时间:2024-07-15 12:55:35浏览次数:15  
标签:arr 排序 temp swapped int 冒泡排序 C语言 解析

文章目录


在这里插入图片描述

在C语言编程中,排序算法是一个非常基础且重要的概念。冒泡排序作为最简单、最易理解的排序算法之一,广泛应用于各种编程教学和实践中。本文将全面解析C语言中的冒泡排序算法,包括其定义、实现、优化方法和性能分析,帮助读者深入理解这一经典算法。

什么是冒泡排序?

冒泡排序(Bubble Sort)是一种简单的排序算法,它通过重复遍历待排序的序列,依次比较相邻元素并交换它们的位置,使较大的元素逐渐“冒泡”到序列的末端。冒泡排序的核心思想是通过不断的比较和交换,将未排序的元素逐步移到正确的位置。

冒泡排序的基本实现

以下是冒泡排序的基本实现代码:

#include <stdio.h>

// 冒泡排序函数
void bubbleSort(int arr[], int n) {
    int i, j, temp;
    for (i = 0; i < n-1; i++) {
        for (j = 0; j < n-i-1; j++) {
            if (arr[j] > arr[j+1]) {
                // 交换arr[j]和arr[j+1]
                temp = arr[j];
                arr[j] = arr[j+1];
                arr[j+1] = temp;
            }
        }
    }
}

// 打印数组函数
void printArray(int arr[], int size) {
    int i;
    for (i = 0; i < size; i++)
        printf("%d ", arr[i]);
    printf("\n");
}

// 主函数
int main() {
    int arr[] = {64, 34, 25, 12, 22, 11, 90};
    int n = sizeof(arr)/sizeof(arr[0]);
    printf("未排序的数组: \n");
    printArray(arr, n);

    bubbleSort(arr, n);
    printf("排序后的数组: \n");
    printArray(arr, n);
    return 0;
}
代码解释
  1. 冒泡排序函数bubbleSort

    • 使用两个嵌套的for循环遍历数组。
    • 内层循环用于比较和交换相邻元素,确保较大的元素逐步移到序列末端。
    • 外层循环用于重复上述过程,直到所有元素都排序完成。
  2. 打印数组函数printArray

    • 遍历数组并打印每个元素,便于查看排序结果。
  3. 主函数main

    • 初始化一个整数数组并计算其大小。
    • 调用bubbleSort函数对数组进行排序。
    • 打印排序前后的数组。
冒泡排序的优化

虽然冒泡排序简单易懂,但其效率较低。以下是几种常见的优化方法:

  1. 标志位优化

    • 使用一个标志位swapped来检测是否发生交换,如果一轮排序中没有发生交换,则说明数组已经有序,可以提前结束排序。

    优化代码示例:

    void bubbleSort(int arr[], int n) {
        int i, j, temp;
        int swapped;
        for (i = 0; i < n-1; i++) {
            swapped = 0;
            for (j = 0; j < n-i-1; j++) {
                if (arr[j] > arr[j+1]) {
                    // 交换arr[j]和arr[j+1]
                    temp = arr[j];
                    arr[j] = arr[j+1];
                    arr[j+1] = temp;
                    swapped = 1;
                }
            }
            // 如果没有发生交换,提前结束排序
            if (swapped == 0)
                break;
        }
    }
    
  2. 双向冒泡排序(鸡尾酒排序)

    • 冒泡排序的另一种改进方法是双向冒泡排序,也称为鸡尾酒排序。该算法在一次遍历中同时从左向右和从右向左进行比较和交换,进一步减少了排序的回合数。

    双向冒泡排序代码示例:

    void cocktailSort(int arr[], int n) {
        int swapped = 1;
        int start = 0;
        int end = n - 1;
        int i, temp;
    
        while (swapped) {
            swapped = 0;
            // 从左向右冒泡
            for (i = start; i < end; ++i) {
                if (arr[i] > arr[i + 1]) {
                    temp = arr[i];
                    arr[i] = arr[i + 1];
                    arr[i + 1] = temp;
                    swapped = 1;
                }
            }
            // 如果没有发生交换,提前结束排序
            if (!swapped)
                break;
            // 减少尾部已排序元素
            --end;
            swapped = 0;
            // 从右向左冒泡
            for (i = end - 1; i >= start; --i) {
                if (arr[i] > arr[i + 1]) {
                    temp = arr[i];
                    arr[i] = arr[i + 1];
                    arr[i + 1] = temp;
                    swapped = 1;
                }
            }
            // 增加头部已排序元素
            ++start;
        }
    }
    
冒泡排序的性能分析

冒泡排序的时间复杂度在最坏情况下为 O ( n 2 ) O(n^2) O(n2),这是因为每次排序都需要比较和交换相邻元素。在最好情况下(当数组已经有序时),时间复杂度为 O ( n ) O(n) O(n),这得益于标志位优化。然而,冒泡排序的平均时间复杂度仍为 O ( n 2 ) O(n^2) O(n2),因此在处理大型数据集时效率较低。

冒泡排序的空间复杂度为 O ( 1 ) O(1) O(1),因为它只需要常数级别的额外空间来存储临时变量。冒泡排序是一个稳定的排序算法,因为相同元素的相对位置不会改变。

冒泡排序的实际应用

虽然冒泡排序在处理大型数据集时效率较低,但它在以下几种情况下仍然有用:

  1. 教学和演示

    • 冒泡排序算法简单易懂,非常适合作为初学者学习排序算法的入门教材。
  2. 小型数据集

    • 在处理小型数据集时,冒泡排序的性能足够,而且实现简单。
  3. 需要稳定排序的场景

    • 在某些需要稳定排序的场景中,冒泡排序可以确保相同元素的相对位置不变。
结论

冒泡排序是C语言中最基础的排序算法之一,其实现简单且易于理解。尽管它的效率不高,但通过标志位优化和双向冒泡排序等方法,可以在一定程度上提升其性能。在学习和使用冒泡排序时,了解其优缺点以及适用场景,能够帮助我们更好地选择和使用排序算法。希望本文能帮助读者深入理解冒泡排序,并在实际编程中灵活应用。

标签:arr,排序,temp,swapped,int,冒泡排序,C语言,解析
From: https://blog.csdn.net/Easonmax/article/details/140435773

相关文章

  • C语言指针超详解——强化篇
    C语言指针系列文章目录入门篇强化篇文章目录C语言指针系列文章目录1.assert断言2.指针的使用和传址调用2.1strlen的模拟实现2.2传值调用和传址调用3.数组名的理解4.使用指针访问数组5.一维数组传参的本质6.冒泡排序7.二级指针8.指针数组9.指针数组模拟......
  • 深度解析:分库分表策略在数据库性能优化中的核心作用
        目录分库分表的核心原理分库(Sharding)分表(Partitioning)综合运用与挑战在探讨分库分表的深度理解之前,先回顾一下为什么数据库系统会面临性能瓶颈。随着互联网业务的飞速发展,数据量呈指数级增长,同时高并发的访问需求对数据库的读写性能提出了更高要求。传统的......
  • 深度解析PHP8 JIT技术:如何助力网站性能飞跃
    本文由ChatMoney团队出品在Web开发领域,提高网站的响应速度一直是开发者和企业所追求的目标。随着技术的不断进步,PHP8的发布为我们带来了一个全新的工具——JIT(Just-In-Time)加速器,它以其独特的优势,成为了提升网站响应速度的重要利器。本文将详细揭秘PHP8的JIT加速器,并探讨其如......
  • 【AI大模型】李彦宏从“卷模型”到“卷应用”的深度解析:卷用户场景卷能给用户解决什么
    文章目录一、理解李彦宏的发言1.1李彦宏的核心观点1.2背景分析二、技术发展:从辨别式到生成式2.1辨别式AI技术2.2生成式AI技术2.3技术发展的挑战三、“卷应用”:聚焦实际应用与价值3.1应用为王3.2技术落地的关键四、“卷场景”:多元化应用场景的探索4.1行业痛点......
  • C语言典型例题
    本系列博客针对于《C程序设计教程(第四版)——谭浩强编著》这本书中的所有例题和习题进行了详细的解释和学习,希望可以对你学习C语言可以有所帮助。有些代码可能会在前面详细解释,后面会一笔带过,希望大家可以多多翻阅,谢谢大家啦!!!嘻嘻!!!//C程序设计教程(第四版)——谭浩强编著//例......
  • 【c语言】你绝对没见过的预处理技巧
    ......
  • 开源缺陷管理系统全解析:如何选择最适合你的?
    国内外主流的10款开源缺陷管理系统软件对比:PingCode、Worktile、Bugzilla、osTicket、MantisBT、Trac、OpenProject、Phabricator、RequestTracker、TheBugGenie。在管理软件项目时,缺陷管理常常是团队面临的一大挑战。选择一个合适的开源缺陷管理系统可以显著提高错误跟踪的......
  • 使用JSONObject构建与解析json对象(简易版)
    构建json实例化一个JSONObject对象,而后调用其put()方法,将数据写入。put()方法的第一个参数为key值,必须为String类型,第二个参数为value,可以为boolean、double、int、long、Object、Map以及Collection等。当然,double以及int等类型只是在Java中,写入到json中时,统一都会以Number类......
  • C语言实现扫雷游戏
    目录一、引言二、游戏规则三、设计思路 1.游戏概述2.数据结构设计3.游戏流程设计4.功能模块划分5.主要算法设计四、游戏设计 1.菜单函数2.主函数3.选择难度函数 4.初始化函数5.布置地雷函数  6.打印函数7.计算雷数函数 8.递归排雷函数9.标记(删除......
  • MySQL存储引擎的选择:深入解析与策略
    MySQL数据库管理系统之所以强大,部分原因在于它提供了多种存储引擎,每种引擎都针对特定的应用场景进行了优化。尽管MySQL支持多种存储引擎,但其中最常用且值得深入探讨的无疑是MyISAM、InnoDB以及MEMORY(HEAP)这三种。每种存储引擎都有其独特的优缺点,合理选择能够显著提升数据库的性......