4. 排序算法

文章目录

  • 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 indexMerge(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.mzph.cn/web/18511.shtml

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

相关文章

js使用链表实现音乐播放器(新增,下一首播放,置顶,删除)

什么是链表 链表是一种线性数据结构&#xff0c;与数组类似&#xff0c;它用于存储一系列元素。不过&#xff0c;与数组在内存中连续存储元素不同&#xff0c;链表中的元素&#xff08;称为节点&#xff09;在内存中可以是非连续存放的。每个节点包含两部分&#xff1a;一部分…

Java开发的saas模式智能制造超强云MES系统源码springboot+mysql+uniapp一整套云MES系统源码

Java开发的saas模式智能制造超强云MES系统源码springbootmysqluniapp一整套云MES系统源码 智能制造超强云MES系统概述&#xff1a; MES以生产车间管理为核心&#xff0c;帮助企业实现生产动态监控和管理。把制造数据管理、计划排程管理、生产调度管理、库存管理、质量管理、人…

swagger-ui页面接口的入参出参与代码实体类不一致有差异、swagger请求实体与预期定义不符、swagger参数与实体中参数不一致原因分析

文章目录 一、问题背景二、问题原因及解决方法 一、问题背景 项目集成swagger之后&#xff0c;发现有个接口的请求跟接口对应不上&#xff0c;把其他接口的请求参数放到当前这个请求上了。 如下图&#xff1a;test1接口的请求参数是其他请求的&#xff0c;并不是test1接口的 …

win10如何查看本机ip地址?三招搞定,快来试试吧

在数字化时代&#xff0c;IP地址作为网络设备的唯一标识&#xff0c;对于计算机使用者来说具有重要意义。无论是为了进行网络设置、远程连接&#xff0c;还是解决网络问题&#xff0c;了解如何查看本机IP地址都是一项必备技能。对于使用Windows 10操作系统的用户来说&#xff0…

简单的 Cython 示例

1&#xff0c; pyx文件 fibonacci.pyx def fibonacci_old(n):if n < 0:return 0elif n 1:return 1else:return fibonacci_old(n-1) fibonacci_old(n-2) 2&#xff0c;setup.py setup.py from setuptools import setup from Cython.Build import cythonizesetup(ext_mod…

node.js(express)+MongoDB快速搭建后端---新手教程

前言&#xff1a; Node.js是一个基于 Chrome V8引擎的JavaScript运行环境&#xff0c;是对于前端工程师来说学习成本最小的后端实现方法&#xff0c;本篇文章总结如何从0-1写一个后端的登录接口 一、检查node环境 先检查自己的node是否安装 一般来说前端工程师的电脑环境肯定…

六面体大米装袋机在提升大米包装效率中的作用

在当今社会&#xff0c;随着科技的飞速发展&#xff0c;各行各业都在寻求创新与突破&#xff0c;以提升生产效率和降低成本。而在大米包装领域&#xff0c;六面体大米装袋机的出现&#xff0c;无疑为整个行业带来了革命性的变化。这种先进的机械设备不仅提高了大米的包装效率&a…

【全开源】沃德校友会管理系统(FastAdmin+ThinkPHP+Uniapp)

一款基于FastAdminThinkPHPUniapp开发的校友会综合服务平台&#xff0c;即校友信息管理平台、活动管理平台、校友服务大厅、校友企业服务平台等&#xff0c;实现集中学校、学院、校友会于一体的基础服务平台的搭建&#xff0c;建设一个满足校友信息化长期发展的、可扩展的综合校…

全面盘点多模态融合算法及应用场景

关注作者&#xff0c;分享AI全维度知识。作者拥有10年互联网服务架构、AI产品研发经验、团队管理经验&#xff0c;同济本复旦硕博&#xff0c;复旦机器人智能实验室成员&#xff0c;阿里云认证的资深架构师&#xff0c;项目管理专业人士&#xff0c;上亿营收AI产品研发负责人 多…

一款更加轻量级的虚拟机:Multipass

一款更加轻量级的虚拟机&#xff1a;Multipass 前言Multipass概述安装Multipassmultipass命令命令使用说明 Multipass的使用查看镜像列表新建和运行虚拟机查看虚拟机列表查看虚拟机信息进入虚拟机外部操作虚拟机删除和释放实例初始化配置虚拟机的调整设置桥接网络接口配置软件源…

Oracle dblink 发现Network 等待事件的分析 enq: KO - fast object checkpoint

所有的sql 通过dblink 查询全部等待中&#xff0c; 同一个SQL 20多个session 在跑&#xff0c;等待事件network&#xff0c;可能怀疑是不是网络断开了&#xff0c;导致没有返回 执行sql 如下&#xff1a; BEGIN Xdblink ; END; 去到dblink 所在的db&#xff0c;发现20多个sql在…

白酒:白酒产地的地域文化与品牌形象

云仓酒庄豪迈白酒&#xff0c;作为中国白酒的一部分&#xff0c;其品牌形象深受产地的地域文化影响。地域文化是一个地区与众不同的文化传统和价值观&#xff0c;它影响着当地人的生活方式和审美观念&#xff0c;进而影响白酒的品牌形象。 首先&#xff0c;白酒产地的历史与传统…

ClickHouse安装教程:开启你的列式数据库之旅

ClickHouse是一个高性能的列式数据库管理系统&#xff0c;适用于在线分析处理&#xff08;OLAP&#xff09;。以下是ClickHouse的一些基本使用步骤&#xff1a; 下载二进制文件&#xff1a;您可以通过运行以下curl命令在Linux、FreeBSD或macOS上本地下载ClickHouse&#xff1a…

Midjourney如何控制光照?提示词灵感来了!

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 Midjourney如何控制光照&#xff1f;提示词灵感来了&#xff01;文章目录 前言总结 前言 Midjourney v6 已经更新好久了&#xff0c;你知道有哪些可以控制光照效果的关键词吗…

全志T527 适配双目tp2815_mipi

一、硬件信息 TP2815&#xff1a; 确认硬件信息&#xff1a; 1、通信接口&#xff1a;TWI2总线&#xff0c;引脚组为PE1 、PE2 2、RESET脚&#xff1a; 二、软件配置 1、设备树 t527 dtsi: bsp/configs/linux-5.15/sun55iw3p1.dtsi t527 uboot-board.dts device/config/chi…

重学java 49 增强for

知之俞明&#xff0c;则行之越笃&#xff1b;行之愈笃&#xff0c;则知之愈益&#xff1b; —— 24.5.28 一、基本使用 1.作用: 遍历集合或者数组 2.格式: for(元素类型 变量名:要遍历的集合名或者数组名) 变量名就是代表的每一个元素 3.快捷键: 集合名或者数组名.for package …

ESXI8.0虚拟机和主机之间进行粘贴复制

1&#xff1a;默认情况下新建一个虚拟机是无法和主机之间进行粘贴复制操作的&#xff0c;主要是为了安全。 2&#xff1a;可以参考下面的文档进行操作&#xff0c;操作成功也只能复制粘贴数据&#xff0c;而无法复制粘贴文件或文件夹 https://knowledge.broadcom.com/externa…

组建RAID后安装系统时发现无法识别硬盘!

计算环境中,RAID(独立磁盘冗余阵列)是一种广泛采用的数据存储技术,它通过组合多个物理硬盘来提升数据读写速度、增加存储容量或提供数据冗余以确保数据安全。然而,用户在使用SAS或SATA RAID阵列卡组建RAID后,可能会遇到在安装操作系统过程中硬盘无法被系统识别的问题。接…

【会议征稿,IEEE出版】第九届信息科学、计算机技术与交通运输国际学术会议(ISCTT 2024,6月28-30)

第九届信息科学、计算机技术与交通运输国际学术会议&#xff08;ISCTT 2024&#xff09;将于2024年6月28-30日在中国绵阳举行。 ISCTT 2024将围绕 “信息科学”、"计算机技术”、“交通运输” 等最新研究领域&#xff0c;为来自国内外高等院校、科学研究所、企事业单位的专…

安卓 view淡入淡出(fade in fade out) kotlin

文章目录 前言一、布局文件二、kotlin扩展方法1.fadeOutAnimation 淡出动画2.fadeInAnimation 淡入动画 三、使用总结 前言 好久没写文章了,简单码一个淡入淡出,我们先上效果图 那么接下来上代码 一、布局文件 我这边直接将activity_main.xml改为下列代码,可以看到其中包含一…