十大排序算法的特点及应用场景

news2024/9/25 21:27:15

一.十大经典排序算法介绍

1. 冒泡排序(Bubble Sort)

原理:通过重复遍历要排序的数列,一次比较两个元素,如果它们的顺序错误就把它们交换过来。遍历数列的工作是重复进行的,直到没有再需要交换的元素为止。

特点:简单但效率低,适合小数据量的排序。

2. 选择排序(Selection Sort)

原理:首先在未排序序列中找到最小(大)元素,存放到排序序列的起始位置,然后,再从剩余未排序元素中继续寻找最小(大)元素,然后放到已排序序列的末尾。以此类推,直到所有元素均排序完毕。

特点:不稳定排序,时间复杂度为O(n^2),但不需要额外的存储空间。

3. 插入排序(Insertion Sort)

原理:通过构建有序序列,对于未排序数据,在已排序序列中从后向前扫描,找到相应位置并插入。

特点:适合少量数据的排序,时间复杂度为O(n^2),但在部分已排序的数据集上效率较高。

4. 希尔排序(Shell Sort)

原理:是插入排序的一种更高效的改进版本。希尔排序通过将原来要排序的数组元素按下标的一定增量分组,对每组使用直接插入排序算法排序;随着增量逐渐减少,每组包含的元素越来越多,当增量减至1时,整个文件恰被分成一组,算法终止。

特点:是第一个突破O(n^2)的排序算法,但时间复杂度依赖于增量序列的选择。

5. 归并排序(Merge Sort)

原理:采用分治法(Divide and Conquer)的一个非常典型的应用。将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。

特点:稳定排序,时间复杂度为O(n log n),但需要额外的存储空间。

6. 快速排序(Quick Sort)

原理:通过一趟排序将待排记录分隔成独立的两部分,其中一部分记录的关键字均比另一部分的关键字小,则可分别对这两部分记录继续进行排序,以达到整个序列有序。

特点:平均时间复杂度为O(n log n),但在最坏情况下(如数组已经有序)时间复杂度为O(n^2)。

7. 堆排序(Heap Sort)

原理:是指利用堆这种数据结构所设计的一种排序算法。堆积是一个近似完全二叉树的结构,并同时满足堆积的性质:即子节点的键值或索引总是小于(或者大于)它的父节点。

特点:不稳定排序,时间复杂度为O(n log n),不需要额外的存储空间。

8. 计数排序(Counting Sort)

原理:不是基于比较的排序算法,其核心在于将输入的数据值转化为键存储在额外开辟的数组空间中。作为一种线性时间复杂度的排序,计数排序要求输入的数据必须是有确定范围的整数。

特点:稳定排序,时间复杂度为O(n+k),其中k是整数的范围。

9. 桶排序(Bucket Sort)

原理:将数组分到有限数量的桶里。每个桶再个别排序(有可能再使用别的排序算法或是以递归方式继续使用桶排序进行排序)。

特点:平均时间复杂度为O(n+k),其中k是桶的数量。

10. 基数排序(Radix Sort)

原理:是一种非比较型整数排序算法,其原理是将整数按位数切割成不同的数字,然后按每个位数分别比较。

特点:稳定排序,时间复杂度为O(nk),其中n是排序元素个数,k是数字位数。

每种排序算法都有其适用的场景和优缺点,选择合适的排序算法可以大大提高程序的效率。

二.算法效率对比

三.综合效率对比建议

1.如果数据量小,用哪种排序方法都可以,效率差别不大

2.如果数据量大;内存充裕建议用快速排序,内存不充裕建议用堆排序。

四.代码示例

1.快速排序c代码

#include <stdio.h>

#include <stdlib.h>

typedef struct _Range {

int start, end;

} Range;

Range new_Range(int s, int e) {

Range r;

r.start = s;

r.end = e;

return r;

}

void swap(int* x, int* y) {

int t = *x;

*x = *y;

*y = t;

}

void quick_sort(int arr[], const int len) {

if (len <= 0)

return; // 避免len等於負值時引發段錯誤(Segment Fault)

// r[]模擬列表,p為數量,r[p++]為push,r[--p]為pop且取得元素

Range r[len];

int p = 0;

r[p++] = new_Range(0, len - 1);

while (p) {

Range range = r[--p];

if (range.start >= range.end)

continue;

int mid = arr[(range.start + range.end) / 2]; // 選取中間點為基準點

int left = range.start, right = range.end;

do {

while (arr[left] < mid) ++left;   // 檢測基準點左側是否符合要求

while (arr[right] > mid) --right; //檢測基準點右側是否符合要求

if (left <= right) {

swap(&arr[left], &arr[right]);

left++;

right--;               // 移動指針以繼續

}

} while (left <= right);

if (range.start < right) r[p++] = new_Range(range.start, right);

if (range.end > left) r[p++] = new_Range(left, range.end);

}

}

int main() {

int arr[] = { 3, 5, 3, 0, 8, 6, 1, 5, 8, 6, 2, 4, 9, 4, 7, 0, 1, 8, 9, 7, 3, 1, 2, 5, 9, 7, 4, 0, 2, 6 };

int len = (int)sizeof(arr) / sizeof(*arr);

quick_sort(arr, len);

int i;

printf("start:\r\n");

for (i = 0; i < len; i++)

printf("%d ", arr[i]);

printf("\n");

return 0;

}

2.堆排序c代码

#include <stdio.h>

#include <stdlib.h>

void swap(int *a, int *b) {

    int temp = *b;

    *b = *a;

    *a = temp;

}

void max_heapify(int arr[], int start, int end) {

    // 建立父節點指標和子節點指標

    int dad = start;

    int son = dad * 2 + 1;

    while (son <= end) { // 若子節點指標在範圍內才做比較

        if (son + 1 <= end && arr[son] < arr[son + 1]) // 先比較兩個子節點大小,選擇最大的

            son++;

        if (arr[dad] > arr[son]) //如果父節點大於子節點代表調整完畢,直接跳出函數

            return;

        else { // 否則交換父子內容再繼續子節點和孫節點比較

            swap(&arr[dad], &arr[son]);

            dad = son;

            son = dad * 2 + 1;

        }

    }

}

void heap_sort(int arr[], int len) {

    int i;

    // 初始化,i從最後一個父節點開始調整

    for (i = len / 2 - 1; i >= 0; i--)

        max_heapify(arr, i, len - 1);

    // 先將第一個元素和已排好元素前一位做交換,再重新調整,直到排序完畢

    for (i = len - 1; i > 0; i--) {

        swap(&arr[0], &arr[i]);

        max_heapify(arr, 0, i - 1);

    }

}

int main() {

    int arr[] = { 3, 5, 3, 0, 8, 6, 1, 5, 8, 6, 2, 4, 9, 4, 7, 0, 1, 8, 9, 7, 3, 1, 2, 5, 9, 7, 4, 0, 2, 6 };

    int len = (int) sizeof(arr) / sizeof(*arr);

    heap_sort(arr, len);

    int i;

    for (i = 0; i < len; i++)

        printf("%d ", arr[i]);

    printf("\n");

    return 0;

}

参考资料:

https://www.runoob.com/w3cnote/ten-sorting-algorithm.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2126052.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

视频编辑SDK解决方案,完整的前端代码+SDK功能交付

繁琐的视频编辑过程往往成为创作者们提升作品质量的瓶颈&#xff0c;美摄科技凭借其深厚的AI技术积累与创新的移动端视频编辑SDK解决方案&#xff0c;正引领着视频编辑领域的新潮流&#xff0c;让每一位创作者都能轻松驾驭创意&#xff0c;实现从灵感闪现到作品呈现的无缝对接。…

点餐小程序实战教程05登录界面搭建

目录 1 设置tab栏2 添加页面3 搭建登录界面3.1 显示头像3.2 显示昵称3.3 注册按钮 总结 上一篇我们讲解了如何在首页加载的时候获取用户信息&#xff0c;一般小程序会在底部放置tab栏&#xff0c;将不同的菜单放置在tab栏中。我们的用户注册逻辑是&#xff0c;如果用户点击了底…

Java码农人生开启手册——多态与重写

一、多态 概念&#xff1a;通俗来说&#xff0c;就是多种形态&#xff1b;具体点&#xff0c;就是去完成某个行为&#xff0c;当不同的对象去完成时会产生不同的状态。 1、多态的实现条件 三个条件&#xff0c;缺一不可&#xff1a; 必须在继承体系下子类必须对父类中方…

训练 Vision Transformer 模型并运行推理

目录 CV Architecture ViT and U-Net Training ViT Florence-2 Load Model Load images CV Scenarios test Genarate CAPTION from the images DENSE REGION CAPTION and REGION_PROPOSA Caption to Phrase Grounding Bounding boxes OCR test Fine Tuning Floren…

2024年转行做网络安全工程师还来得及吗?薪资怎么样呢

&#x1f91f; 基于入门网络安全/黑客打造的&#xff1a;&#x1f449;黑客&网络安全入门&进阶学习资源包 2022年以来&#xff0c;我国网络安全行业的市场规模持续增长&#xff0c;根据市场调研在线网发布的2023-2029年中国网络安全集成行业市场运行态势及发展趋向分析…

Pycharm配置ssh远程服务器解析器

算法学习、4对1辅导、论文辅导或核心期刊可以通过公众号滴滴我 文章目录 需求配置流程 需求 之前在开发中&#xff0c;Pycharm都是通过本机Python环境来解析。但有时候&#xff0c;可能受限于本机电脑配置原因&#xff0c;导致运行速度并不快。因此推荐大家尝试下&#xff0c…

一文带你彻底掌握二分查找

1. 认识二分查找 二分查找也被称为折半查找&#xff0c;他是一种查询效率较高的查找方式&#xff0c;普通查找的方式通常是从头到尾遍历一遍数组&#xff0c;二分查找的方式是找到数组中间的那个元素mid与目标值target进行比较&#xff0c;比target小就去前半段找&#xff0c;…

Java中List集合去重

反问问题&#xff1a;为什么不直接使用 Set 或者 LinkedHashSet 呢 实际场景&#xff1a;实际的业务开发中遇到的情况会更复杂。比如&#xff0c;List 集合可能是历史遗留问题&#xff0c;也有可能是调用接口返回的类型限制&#xff0c;只能使用 List 接收&#xff0c;又或者是…

Qualcomm Linux 交叉编译应用程序

1. 前提条件 Ubuntu 20.04 系统 Qualcomm RB3 Gen2开发板 2.下载并安装 eSDK 平台 1.从 Qualcomm 发布存档平台下载 eSDK。 wget https://artifacts.codelinaro.org/artifactory/qli-ci/flashable-binaries/qimpsdk/qcm6490/x86/qcom-6.6.28-QLI.1.1-Ver.1.1_qim-product-s…

消除数字球-第15届蓝桥省赛Scratch初级组真题第5题

[导读]&#xff1a;超平老师的《Scratch蓝桥杯真题解析100讲》已经全部完成&#xff0c;后续会不定期解读蓝桥杯真题&#xff0c;这是Scratch蓝桥杯真题解析第184讲。 如果想持续关注Scratch蓝桥真题解读&#xff0c;可以点击《Scratch蓝桥杯历年真题》并订阅合集&#xff0c;…

python程序使用nohup后台执行不能实时输出到定向文件的解决方法

问题描述&#xff1a;使用nohup命令后台执行python&#xff0c;但python中print方法打印结果不能实时输出到nohup后台定向文件&#xff0c;只能在程序结束时一次性输出。典型问题样例&#xff1a;在python中使用了os.system(command)方法&#xff0c;command命令打印的结果可以…

免费爬虫软件“HyperlinkCollector超链采集器v0.1”

HyperlinkCollector超链采集器单机版v0.1 软件采用python的pyside2和selenium开发,暂时只支持window环境&#xff0c;抓取方式支持普通程序抓取和selenium模拟浏览器抓取。软件遵守robots协议。 首先下载后解压缩&#xff0c;然后运行app目录下的HyperlinkCollector.exe 运行…

网页与App无缝衔接,揭秘拉起应用的黑科技!

随着移动互联网的飞速发展&#xff0c;App已经成为了我们日常生活中不可或缺的一部分。然而&#xff0c;在推广和运营App的过程中&#xff0c;如何让用户更便捷地从网页跳转到App&#xff0c;一直是困扰推广者的难题。今天&#xff0c;我们就来聊聊网页拉起应用这一黑科技&…

开源 AI 智能名片 S2B2C 商城小程序中的全渠道供应策略

摘要&#xff1a;本文深入探讨在开源 AI 智能名片 S2B2C 商城小程序的情境下&#xff0c;全渠道供应的运行机制。阐述各环节企业相互配合的重要性&#xff0c;重点分析零售企业在其中的关键作用&#xff0c;包括协调工作、信息传递、需求把握等方面&#xff0c;旨在实现高效的全…

Python中的上下文管理器:提升代码的优雅与安全

在编写Python程序时&#xff0c;处理资源&#xff08;如文件、网络连接、数据库会话等&#xff09;的正确打开和关闭至关重要。不当的资源管理可能导致内存泄漏、数据损坏等问题。幸运的是&#xff0c;Python提供了一种优雅的方式来解决这个问题——上下文管理器。本文将探讨上…

象过河轮胎进销存,轻松管理进出库以及废旧轮胎回收

在轮胎行业&#xff0c;高效的进销存管理与废旧轮胎的回收是两大核心挑战&#xff0c;象过河轮胎进销存应运而生。软件专为轮胎行业量身定制&#xff0c;从进货&#xff0c;销售&#xff0c;到库存盘点&#xff0c;财务&#xff0c;轮胎废旧回收等一体化管理&#xff0c;以科技…

【每日刷题】Day118

【每日刷题】Day118 &#x1f955;个人主页&#xff1a;开敲&#x1f349; &#x1f525;所属专栏&#xff1a;每日刷题&#x1f34d; &#x1f33c;文章目录&#x1f33c; 1. 123. 买卖股票的最佳时机 III - 力扣&#xff08;LeetCode&#xff09; 2. 188. 买卖股票的最佳时…

如何用python打开csv文件路径

python读取CSV文件方法&#xff1a; 方法1&#xff1a;可先用以下代码查看当前工作路径&#xff0c;然后将CSV文件放在该路径下。 import os os.getcwd() 方法2&#xff1a;&#xff08;绝对路径&#xff09; import pandas as pd iris_trainpd.read_csv(E:\Study\DataSets\ir…

树莓派最强大的应用商店,你知道嘛?

Pi-Apps —— Raspberry Pi 最强大的应用商店 不久前&#xff0c;我获得了我的第一台Raspberry Pi。和大多数从Windows转移到Linux&#xff08;特别是Raspberry Pi用户&#xff09;的用户一样&#xff0c;我被这些问题困扰得快要疯了&#xff1a; 如何在我的Raspberry Pi上安装…

vue3 响应式API customRef()

使用ref()定义响应式数据&#xff1a; <template><div><div>{{ inputValue }}</div><input type"text" v-model"inputValue"></div> </template> <script setup lang"ts"> import { ref } fro…