4. 排序算法

news2025/1/17 16:05:03

文章目录

  • 1.简单排序
    • 1.1 冒泡排序
      • 1.1.1 步骤+核心思想
      • 1.1.2 参考代码
      • 1.1.3 时间复杂度
      • 1.1.4 空间复杂度
      • 1.1.5 优化
    • 1. 2. 选择排序
      • 1.2.1 核心思想
      • 1.2.2 步骤
      • 1.2.3 参考代码
      • 1.2.4 时间复杂度
      • 1.2.5 空间复杂度
      • 1.2.6 优化
    • 1.3 插入排序
      • 1.3.1 思想
      • 1.3.2 步骤
      • 1.3.3 参考代码
      • 1.3.4 时间复杂度
      • 1.3.5 空间复杂度
      • 1.3.6 优化
    • 1.4 分析
    • 1.5 分析
    • 1.6 小结
  • 2.复杂排序
    • 2.1. 归并排序
      • 2. 1.1 思想
      • 2. 1.2 步骤
      • 2. 1.3 参考代码
      • 2. 1.4 时间复杂度
      • 2. 1.5 空间复杂度
      • 2. 1.6 迭代归并排序
      • 2. 1.7 练习
    • 2. 2 快速排序
      • 2. 2.1 思想
      • 2. 2.2 步骤
      • 2. 2.3 参考代码
      • 2. 2.4 时间复杂度
      • 2. 2.5 空间复杂度
      • 2. 2.6 优化
    • 2.3. 希尔排序
      • 2.3.1 算法原理
    • 2.3.2 步骤
      • 2.3.3 参考代码
      • 2.3.4 比较:插入排序与希尔排序
      • 2.3.5 优化
    • 2.4 小结
  • 3算法选择标准
  • 4 练习

1.简单排序

1.1 冒泡排序

1.1.1 步骤+核心思想

思想:(1)执行一遍核心算法:
需要把最大的数一直挪到最后面;第一遍排序 i< n-1;交换的是arr+i+1;
(2)挪动n-1轮;(n --1 )–1

步骤:
(1)首先实现一趟冒泡

void bubble(int arr[],int n);

(2)再实现多趟冒泡

void bubble_sort(int arr[],int n);

1.1.2 参考代码

void bubble(int arr[],int n){
    for(int i=0;i<n-1;++i){
        if(arr[i] > arr[i+1]){
            swap(arr+i,arr+i+1);
        }
    }
}
void bubble_sort(int arr[],int n){
    for(int i=n;i>1;i--){
        bubble(arr,i);
    }
    // while(n>1) bubble(arr,n--);
}

1.1.3 时间复杂度

一共比较次,即

1.1.4 空间复杂度

随着的增长,排序不需要增加额外空间,空间复杂度为o(1)。

1.1.5 优化

尝试对已排序的数列冒泡排序

1. 2. 选择排序

1.2.1 核心思想

找出最大数字和索引,和最后的数字进行交换; /索引1–n;
仍是从前往后,把最大的往后放;.//交换次数: n-1; arr+i-1 ;–1

1.2.2 步骤

首先实现一趟,找出最大数字的下标

int find_max_index(int arr[],int n);

实现多趟将最大数字与最后数字交换

int selection_sort(int arr[],int n);

1.2.3 参考代码

int find_max_index(int arr[],int n){
    int max = arr[0];
    int max_index = 0;
    for(int i=1;i<n;++i){
        if(max < arr[i]){
            max = arr[i];
            max_index = i;
        }
    }
    return max_index;
}
void selection_sort(int arr[],int n){
    for(int i=n;i>1;--i){
        int max_index = find_max_index(arr,i);
        swap(arr+max_index,arr+i-1);// i是数组长度,最后一个元素下标要减1
    }
}

1.2.4 时间复杂度

一共比较次,即

1.2.5 空间复杂度

随着的增长,排序不需要增加额外空间,空间复杂度为0(1)。

1.2.6 优化

同时选择最大值和最小值

1.3 插入排序

1.3.1 思想

最小的往前面放; 最后一个元素是待插入元素 逆序排;i+1
插入n-2次

1.3.2 步骤

实现向有序数列插入一个数字;

void insert(int arr[],int n);

依次在数组中插入数字

void insertion_sort(int arr[],int n);

1.3.3 参考代码

void insert(int arr[],int n){
    for(int i=n-2;i>=0;--i){
        if(arr[i+1]<arr[i]){
            swap(arr+i+1,arr+i);
        }
    }
}
void insertion_sort(int arr[],int n){
    for(int i=2;i<=n;++i){
        insert(arr,i);
    }
}

1.3.4 时间复杂度

一共比较在这里插入图片描述
次,即o(n^2)

1.3.5 空间复杂度

随着的增长,排序不需要增加额外空间,空间复杂度为o(1)。

1.3.6 优化

用移动代替交换

void insert(int arr[],int n){ // 最后一个元素是待插入元素
    int key=arr[n-1];
    int i=0;
    for(i=n-1;i>=1&&arr[i-1]>key;--i){
        arr[i] = arr[i-1];
    }
    arr[i] = key;
}

void insertion_sort(int arr[],int n){
    for(int i=1;i<n;++i){// 插入元素i操作
        insert(arr,i+1); // i+1表示数组长度,最后一个元素表示要插入的
    }
}

1.4 分析

试一下下面的数据比较次数和交换次数

随机
正序
逆序

1.5 分析

使用swap()和cmp()替换排序函数中的交换和比较处理。
实现向qsort一样的排序函数。
例如:
void bubble_sort(void* arr[],int n,int size,int (cmp)(const void,const void*));

1.6 小结

排序的稳定性
在待排序的序列中,存在多个相同的关键字记录,经过排序这些记录相对位置保持不变,则这种排序称为稳定的,否则称为不稳定的。

No算法Stable
1冒泡排序 Bubble SortYes
2插入排序 Insertion SortYes
3选择排序 Selection SortNo

2.复杂排序

2.1. 归并排序

2. 1.1 思想

在这里插入图片描述

2. 1.2 步骤

实现两个有序数组的合并

void merge(int arr[],int n,int mid);

拆分并合并数组

void merge_sort(int arr[],int n);

2. 1.3 参考代码

void merge(int arr[],int n,int mid){
    int temp[n];
    memcpy(temp,arr,n*sizeof(int));
    int p = 0,q = mid,k = 0;
    while(p<mid && q<n) arr[k++] = temp[p]<=temp[q]?temp[p++]:temp[q++];
    if(p<mid) memcpy(arr+k,temp+p,(mid-p)*sizeof(int));
    if(q<n) memcpy(arr+k,temp+q,(n-q)*sizeof(int));
}
void merge_sort(int arr[],int n){
    if(n <= 1) return;
    int mid = n/2;
    merge_sort(arr,mid);
    merge_sort(arr+mid,n-mid);
    merge(arr,n,mid);
}

2. 1.4 时间复杂度

一共拆分次,每次比较个元素,一共比较次。

2. 1.5 空间复杂度

随着的增长,排序需要增加额外空间(临时数组和递归调用函数栈),空间复杂度为。

2. 1.6 迭代归并排序

void Merge(int* arr,int n,int mid){
     int res[n];
     int i=0,j=mid,k=0;
     while(i<mid && j<n){
         res[k++] = (arr[i]<arr[j])?arr[i++]:arr[j++];
     }
     if(i<mid) memcpy(res+k,arr+i,(mid-i)*sizeof(int));
     if(j<n) memcpy(res+k,arr+j,(n-j)*sizeof(int));
     memcpy(arr,res,n*sizeof(int));
}
void SubMerge(int* arr,int n,int step){
    int len = n;
    for(int i=0;i+step<n;i+=step*2){ // merge index
    Merge(arr+i,min(2*step,len),step);
    len -= 2*step;
    }
}
void MergeSort(int* arr,int n){
     for(int i=1;i<n;i*=2){
         SubMerge(arr,n,i);
     }
}

2. 1.7 练习

剑指 Offer 51. 数组中的逆序对:
https://www.jianshu.com/go-wild?ac=2&url=https%3A%2F%2Fleetcode-cn.com%2Fproblems%2Fshu-zu-zhong-de-ni-xu-dui-lcof%2F

2. 2 快速排序

2. 2.1 思想

左游标是i哨兵,右游标是j哨兵。
右边发现比基准小的数,左边发现比基准大的数,进行交换;
到最后相遇的时候,基准进行交换;
在这里插入图片描述

第一次交换

在这里插入图片描述

第二次交换

在这里插入图片描述

基准交换

在这里插入图片描述

在这里插入图片描述

2. 2.2 步骤

根据基准元素重排数组

int partition(int arr[],int n);

依次排列两个部分

void quick_sort(int arr[],int n);

2. 2.3 参考代码

int partition(int arr[],int n){
    int key = arr[0];
    int p = 0,q = n-1;
    while(p<q){
        while(p<q && arr[q]>=key) q--;
        arr[p] = arr[q];
        while(p<q && arr[p]<=key) p++;
        arr[q] = arr[p];
    }
    arr[p] = key;
    return p;
}
void quick_sort(int arr[],int n){
    if(n<=1) return;
    int pivot = partition(arr,n);
    quick_sort(arr,pivot);
    quick_sort(arr+pivot+1,n-pivot-1);
}

2. 2.4 时间复杂度

一共拆分次,每次比较个元素,一共比较次。

2. 2.5 空间复杂度

随着的增长,排序需要增加额外空间(递归函数栈空间),空间复杂度为。

2. 2.6 优化

交换指针法

int partition(int arr[],int n){
    int key = arr[0];
    int p = 0,q = n-1;
    while(p<q){
        while(p<q && arr[q]>=key) q--;
        while(p<q && arr[p]<=key) p++;
        if(p>=q) break;
        swap(arr+q,arr+p);
    }
    swap(arr,arr+q);
    return q;
}

2.3. 希尔排序

2.3.1 算法原理

给定一个长度n的列表,选择一定的步长gap,将列表分成若干个子列表sublist。
例如:长度n=9步长gap=3分成3个子列表sublist。

对每一个子列表sublist进行插入排序。
即:arr[i],arr[i+gap];
依次减小步长gap,重复上述操作。直到gap为1。
希尔排序比插入排序的优势:
通过分组排序使元素逐渐靠近最终位置,从而减少了插入排序 时的移动次数。(先粗调再微调)
在这里插入图片描述

2.3.2 步骤

嵌套关系
(1)根据间距gap插入

void insert(int arr[],int n,int gap);

(2)根据间距gap执行插入排序

void insertion_sort(int arr[],int n,int gap);

(3) 划分间距gap并执行排序

void shell_sort(int arr[],int n);

2.3.3 参考代码

void insert(int arr[],int n,int gap){
    for(int i=n-1-gap;i>=0;i-=gap){
        if(arr[i+gap]<arr[i]){
            swap(arr+i+gap,arr+i);
        }else{
            break;
        }
    }
}
void insertion_sort(int arr[],int n,int gap){
    for(int i=gap;i<=n;++i){
        insert(arr,i,gap);
    }
}
void shell_sort(int arr[],int n){
    int gap = n;
    do{
        gap = gap/2;
        insertion_sort(arr,n,gap);
    }while(gap>1);
}

2.3.4 比较:插入排序与希尔排序

>

希尔排序比插入排序更加高效,其时间复杂度和空间复杂度均优于插入排序。然而,由于希尔排序是一种不稳定的排序算法,无法保证相等元素的相对位置不变,所以在某些情况下可能不适用。

2.3.5 优化

移动代替交换

void insert(int arr[],int n,int gap){   
    int key=arr[n-1];
    int i=0;
    for(i=n-1;i>=gap&&arr[i-gap]>key;i-=gap){
        arr[i] = arr[i-gap];
    }
    arr[i] = key;
}

void insertion_sort(int arr[],int n,int gap){
    for(int i=gap;i<n;++i){// 插入元素i操作
        insert(arr,i+1,gap);
    }
}

void shell_sort(int arr[],int n){
    for(int i=n/2;i>0;i/=2){// gap序列
        insertion_sort(arr,n,i);
    }
}

2.4 小结

No算法AlgorithmTime ComplexitySpace ComplexityStable
1快速排序QuicksortO(nlog2 n)O(nlog2 n)No
2归并排序 MergesortO(nlog2 n)O(n)Yes
3希尔排序 Shell SortO(nlog2 n)~O(n^2)O(1)No

3算法选择标准

如何选择排序算法?(定性)

No.准则排序算法
1很少的元素插入排序
2几乎有序的元素插入排序
3关注最坏的情况堆排序
4希望能够得到一个好的平均情况下性能快速排序
5元素是从一个密集集合中抽取出桶排序
6希望尽可能少的写代码插入排序

4 练习

可以保持在原有时间复杂度前提下对双向链表进行排序的算法有:(不定项)

堆排序
并归排序
快速排序
希尔排序
可以保持在原有时间复杂度前提下对单向链表进行排序的算法有:(不定项)

选择排序
插入排序
快速排序
冒泡排序

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

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

相关文章

工控一体机10.1寸显示器电容触摸屏(YA1308101JK)产品规格说明书

如果您对工控一体机有任何疑问或需求&#xff0c;或者对如何集成工控一体机到您的业务感兴趣&#xff0c;可移步控芯捷科技。 一、硬件功能介绍 YA1308101JK产品介绍&#xff1a; YA1308101JK搭载 Android10 主流操作系统&#xff0c;具有系统版本更高、占用内存更低、运行效率…

Orcle查询组合字段重复的数据

oracle拼接字符串 在Oracle中&#xff0c;可以使用||运算符或CONCAT函数来拼接字符串。 使用||运算符&#xff1a; SELECT Hello, || World! AS concatenated_string FROM dual;使用CONCAT函数&#xff1a; SELECT CONCAT(Hello, , World!) AS concatenated_string FROM d…

Python的解析网页【XPath】

XPath 什么是XPath XPath&#xff08;XML Path Language&#xff09;是一种用于在XML文档中定位和选择节点的语言。它是W3C&#xff08;World Wide Web Consortium&#xff09;定义的一种标准查询语言&#xff0c;广泛用于解析和操作XML文档。 安装配置 安装lxml&#xff1…

微软Build开发者大会速览,OpenAI CEO站台剧透新模型

5月22日凌晨&#xff0c;微软Build 2024开发者大会在美国西雅图召开。微软CEO萨蒂亚纳德拉在会上发表主题演讲&#xff0c;宣布了超过50项产品更新&#xff0c;涵盖AI基础设施、模型产品以及生产力工具等多个领域。纳德拉强调&#xff0c;微软一直致力于让人工智能理解人类并帮…

AI工具助力:打造完美答辩PPT的秘诀

很多快要毕业的同学在做答辩PPT的时候总是感觉毫无思路&#xff0c;一窍不通。但这并不是你们的错&#xff0c;对于平时没接触过相关方面&#xff0c;第一次搞答辩PPT的人来说&#xff0c;这是很正常的一件事。一个好的答辩PPT可以根据以下分为以下几部分来写。 1.研究的背景和…

时间|基于SprinBoot+vue的时间管理系统(源码+数据库+文档)

时间管理系统 目录 基于SprinBootvue的时间管理系统 一、前言 二、系统设计 三、系统功能设计 1管理员功能模块 2用户功能模块 四、数据库设计 五、核心代码 六、论文参考 七、最新计算机毕设选题推荐 八、源码获取&#xff1a; 博主介绍&#xff1a;✌️大厂码农…

Unity版本使用情况统计(更新至2024年4月)

UWA发布&#xff5c;本期UWA发布的内容是第十四期Unity版本使用统计&#xff0c;统计周期为2023年11月至2024年4月&#xff0c;数据来源于UWA网站&#xff08;www.uwa4d.com&#xff09;性能诊断提测的项目。希望给Unity开发者提供相关的行业趋势作为参考。 2023年11月 - 2024年…

ChatGPT实现法语口语练习APP

使用ChatGPT实现一个法语口语练习APP可以提供一个强大的工具&#xff0c;帮助学习者提高他们的口语能力。以下是一个详细的实现流程&#xff0c;包括需求分析、技术选型、开发、测试和部署。北京木奇移动技术有限公司&#xff0c;专业的软件外包开发公司&#xff0c;欢迎交流合…

考研数学|张宇《1000题》正确率达到多少算合格?

张宇1000题是考研数学难度较大一些的题集&#xff0c;题目难&#xff0c;计算量大&#xff0c;第一次做题时&#xff0c;如果正确率能达到60%&#xff0c;那就已经算是不错了。有些题目设计得挺巧妙&#xff0c;有时候第一次见&#xff0c;想不出解题思路也是正常的。 传统习题…

C51单片机开发--库函数

知不足而奋进 望远山而前行 目录 系列文章目录 文章目录 前言 目标 内容 开发过程回顾 使用库函数点灯 什么是库函数? 面向库函数和面向寄存器开发 使用delay模块延时 总结 前言 在嵌入式系统开发中&#xff0c;使用库函数是提高开发效率、简化编程的重要手段之一…

文章解读与仿真程序复现思路——电力系统自动化EI\CSCD\北大核心《基于优先指数的配电网分布式储能序次规划》

本专栏栏目提供文章与程序复现思路&#xff0c;具体已有的论文与论文源程序可翻阅本博主免费的专栏栏目《论文与完整程序》 论文与完整源程序_电网论文源程序的博客-CSDN博客https://blog.csdn.net/liang674027206/category_12531414.html 电网论文源程序-CSDN博客电网论文源…

我国赤泥年产量庞大 政策引导下赤泥绿色利用率将不断提升

我国赤泥年产量庞大 政策引导下赤泥绿色利用率将不断提升 赤泥是指从铝土矿中提炼氧化铝后产生的强碱性工业固体废渣&#xff0c;由于含大量氧化铁&#xff0c;表面呈现红色&#xff0c;而得名赤泥。   赤泥通常包含氧化铝、氧化铁、二氧化硅、氧化钙、碱金属及其他微量元素&…

【深度 Q 学习-01】 Q学习概念和python实现

文章目录 一、说明二、深度 Q 学习概念三、python实现四、结论 关键词&#xff1a;Deep Q-Networks 一、说明 在强化学习 &#xff08;RL&#xff09; 中&#xff0c;Q 学习是一种基础算法&#xff0c;它通过学习策略来最大化累积奖励&#xff0c;从而帮助智能体导航其环境。它…

SAP实施方法论

SAP新实施方法论 RDS Activate

分布式拒绝服务解决方式

在网络安全领域中&#xff0c;分布式拒绝服务&#xff08;DDoS&#xff09;攻击始终占据着举足轻重的地位&#xff0c;其影响力不容忽视。随着网络技术的日新月异和网络环境的日益复杂化&#xff0c;DDoS攻击不仅变得愈发频繁&#xff0c;而且其破坏性和影响力也呈现出惊人的增…

二区5分纯生信|单细胞+非负矩阵+AlphaFold+机器学习组合

说在前面 学文不看刊 这篇分析总体来说工作量不算大&#xff0c;scRNA联合bulkRNA分析&#xff0c;多种机器学习组合预测模型&#xff0c;最后用了AlphaFold2预测蛋白及AutoDock分子对接 非常适合小白学习和模仿&#xff0c;其次在筛选出核心基因后可以再加几个外部数据集观…

【最优化方法】实验四 约束最优化方法的MATLAB实现

实验的目的和要求&#xff1a;通过本次实验使学生较为熟练使用MATLAB软件&#xff0c;并能利用该软件进行约束最优化方法的计算。 实验内容&#xff1a; &#xff11;、罚函数法的MATLAB实现 &#xff12;、可行方向法的MATLAB实现 学习建议&#xff1a; 本次实验就是要通…

抠像标签合并到原图,jpg 和 png合并,查看标签是否准确

抠像 原图 和 标签合并&#xff0c;查看抠像是否准确 合并后的图&#xff0c;是带有 羽化 效果的 import osimport cv2 import numpy as npdef apply_mask_with_feathering(original_image_path, mask_image_path):# 读取原图和mask图original_image cv2.imread(original_im…

一分钟教你学浪app视频怎么缓存

你是否在学浪app上苦苦寻找如何缓存视频的方法&#xff1f;你是否想快速、轻松地观看自己喜欢的视频内容&#xff1f;那么&#xff0c;让我们一起探索一分钟教你如何缓存学浪app视频的技巧吧&#xff01; 学浪下载工具我已经打包好了&#xff0c;有需要的自己下载一下 学浪下…

豆瓣内容抓取:使用R、httr和XML库的完整教程

概述 在数据分析和统计领域&#xff0c;R语言以其强大的数据处理能力和丰富的包库资源而闻名。它不仅提供了一个灵活的编程环境&#xff0c;还拥有专门用于数据抓取和处理的工具&#xff0c;如httr和XML库。这些工具使得从各种网站上抓取数据变得简单而高效。 豆瓣网站作为一个…