深入理解快速排序

一、快速排序

        快速排序是冒泡排序的一种改进算法,相比于冒泡排序效率更优。

算法过程分析:

        通过采用分治策略,围绕一个 x 将原始数组划分为两个子数组,使得前一个子数组的元素≤ x ≤ 后一个子数组元素,对两个子数组进行递归排序,再合并成一个有序数组。

        1.选取一个基准元素 key(通常默认为数组的最左端),通过 key 将数组分为左右两端,使得左端数组全部 ≤ 基准元素 key ,右端数组全部元素 ≥ 基准元素key。

        2.两端数组可以独立排序。对于左端数组,又可以取一个基准元素,将该端数组元素分成左右两部分,使得左端数组全部 ≤ 基准元素 key ,右端数组全部元素 ≥ 基准元素key。右侧数组做类似处理。

        3.通过递归重复上述操作,当全部递归结束时,原数组也排序完成。

 细节分析:

        步骤1:将数组如何分成两个子数组,使得左端数组全部 ≤ 基准元素 key ,右端数组全部元素 ≥ 基准元素key

①j从右向左,找到小于key的数组元素

②i从左向右,找到大于key的数组元素,将二者交换。

③上述操作直至 i==j 时结束 

④此时再将基准值与 a[i](或a[j],i==j)交换,就使得两个子数组上述条件成立。

int i=left,j=right;
while(i!=j)
{while(a[j]>=temp && i<j) j--;while(a[i]<=temp && i<j) i++;swap(a[i],a[j]);
}
swap(a[left],a[i]); 

         步骤2:

分治策略,将左右两个子数组进行相同的操作。

quick_sort(left,i-1);
quick_sort(i+1,right);

 重复操作,通过递归分别进入两个子数组操作。

当全部返回时,结束递归,数组完成排序。

        思考:为什么 i 从左往右移动,j从右往左移动时,j 先移动?

  因为当 i 先移动时,i 一定会往右至少移动一次,出现以下情况:

① i 不停移动,直至移动到数组右端,此时交换i,j,出现错误,如图。

② i 先移动,移动到中间某个点时,j再移动,移动到i,j相等时交换,出现错误。

相反,如果是 j 先移动, 都能使得正确交换,必须先找到小于基准值key的元素

二、快速排序完整代码

#include<stdio.h>
#include<string.h>
#include<stdlib.h>
int a[10001];
int n;
void swap(int a1,int b1)    {int w=a[a1];a[a1]=a[b1];a[b1]=w;}  //交换函数
void qsort1(int begin,int end) // 快排的实现
{if(begin > end)  return ;  // 退出快排函数 避免出现死循环int tem=a[begin];    // 记录基准点int i=begin,j=end;while(i!=j){while(a[j]>=tem && i<j)   j--;  //从右边开始找小于基准点的数while(a[i]<=tem && i<j)   i++;  //从左边开始找大于基准点的数if(i<j) swap(i,j);    }a[begin]=a[i]; // 交换基准点和第i个 确保基准点左边全部比其小 右边全部不它大a[i]=tem;qsort1(begin,i-1);  // 继续分成两个部分进行快速排序qsort1(i+1,end);}
int main()
{scanf("%d",&n);for(int i=1;i<=n;i++)scanf("%d",&a[i]);qsort1(1,n);for(int i=1;i<=n;i++) printf("%d ",a[i]);// system("pause");return 0;
}

三、算法的性能分析

时间复杂度:

        理想情况:每次都尽可能将左端数组和右端数组等分递归,这样通过计算得出

                T(n)=2T(\frac{n}{2})+\Theta (n)

           时间复杂度与归并排序相同,为  O(nlogn) 

        最差情况:不难发现,当排序数组为顺序或者逆序时,每次 i,j 的移动都是从左到右

                时间计算 会与冒泡排序相同        复杂度为 O(n^{2})

            一般情况下,快速排序的时间复杂度为 O(nlogn),但是不稳定。

  洛谷—排序P1177  这道题,数据仍然卡快速排序,时间复杂度来到  O(n^{2}) ,需要对其进行优化。

四、快速排序优化

         上述快速排序总是将基准值默认为数组中第一个元素,当数组成顺序或者逆序时,出现最慢的情况。

        优化方法:在排序数组中随机选择一个数作为基准数进行排序。

基准数随机的结果:

         ①运行时间与输入数据的次序无关

        ②无特定输入数据匹配最差情况

        ③最差情况仅仅由随机数生成器所决定

对随机化快速排序的时间复杂度:O(nlogn)  (不稳定)

四、sort() 函数

        在C++中使用sort()函数需要使用#include<algorithm>头文件。algorithm意为"算法",是C++的标准模版库(STL)中最重要的头文件之一,提供了大量基于迭代器的非成员模版函数。

 可以简单使用 sort() 函数对数组进行排序

#include<stdio.h>
#include<string.h>
#include<algorithm>
int main()
{int n,a[10001];scanf("%d",&n);for(int i=1;i<=n;i++) scanf("%d",&a[i]);sort(a+1,a+1+n); // 使数组从  a[1] 到 a[n] 排序for(int i=1;i<=n;i++) printf("%d ",a[i]);return 0;
}

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

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

相关文章

WebGIS开发0基础必看教程:鹰眼的实现思路

1.背景 鹰眼功能是WebGIS中的一种常见功能&#xff0c;利用一些开源的框架实现起来非常方便。当然&#xff0c;按照博主从底层谈WebGIS这套系列的风格&#xff0c;在这里还是跟大家一起探讨探讨不借助第三方框架开发鹰眼的简单实现思路。 2.鹰眼功能分析 3.实现思路设计 3.1…

C语言 - 堆栈二叉树的基础及应用

1.历史 堆栈是为了减少程序内存占用的问题而发明的 机器上电后&#xff0c;所有的变量都需要copy到内存中运行&#xff0c;但是机器的内存大小一直都是比较有限的&#xff0c;所以堆栈和局部变量两个概念被提出来 2.栈 每次手动创建、删除变量非常麻烦&#xff0c;于是…

【spring】@Bean注解学习

Bean介绍 Bean用于指示一个方法应该产生一个Bean对象&#xff0c;并将其交给Spring容器管理。 当使用Bean注解的方法被Spring容器调用时&#xff0c;它只会执行一次&#xff0c;随后该方法返回的Bean对象会被添加到Spring的IOC&#xff08;Inversion of Control&#xff0c;控…

静电ESD整改:原因、影响与解决方案详解?|深圳比创达电子

静电&#xff08;ESD&#xff09;是在日常生活和工作中常见的现象&#xff0c;但它可能对电子设备和器件造成严重的损坏。本文将介绍静电ESD的定义、原因、影响以及解决方案&#xff0c;帮助大家更好地了解ESD问题&#xff0c;并采取相应的整改措施。 一、静电ESD的定义 静电…

L2-4 寻宝图(Python3)

给定一幅地图&#xff0c;其中有水域&#xff0c;有陆地。被水域完全环绕的陆地是岛屿。有些岛屿上埋藏有宝藏&#xff0c;这些有宝藏的点也被标记出来了。本题就请你统计一下&#xff0c;给定的地图上一共有多少岛屿&#xff0c;其中有多少是有宝藏的岛屿。 输入格式&#xf…

Python Excel 文本编辑库之xlsxwriter使用详解

概要 在现代数据处理和报表生成中,Excel 文件是一个非常常见的格式。Python XlsxWriter 库是一个强大的工具,可以帮助开发者轻松创建和编辑 Excel 文件,并且具有高度的灵活性和可定制性。本文将全面介绍 XlsxWriter 库的原理、功能、用法,并通过丰富的示例代码来展示其强大…

什么是工业数采网关?如何远程数采?

随着工业自动化的不断发展&#xff0c;数据采集与远程控制成为了生产过程中不可或缺的一环。而工业数采网关&#xff0c;作为连接工业设备与上位管理系统之间的桥梁&#xff0c;扮演着越来越重要的角色。HiWoo Box 作为一款优秀的工业数采网关产品&#xff0c;为 PLC、传感器、…

蓝牙系列十五:协议栈GAP层分析

Generic Access Profile&#xff08;通用访问规范&#xff09; 它在用来控制设备连接和广播&#xff0c;用于提供蓝牙设备的通用访问功能&#xff0c;包括设备发现、连接、鉴权、服务发现等等。 GATT是建立连接后通信规范&#xff0c; 而蓝牙是通过GAP建立通信的。 GAP 使你的…

边缘智能融合区块链:研究现状、应用及挑战

源自&#xff1a;信息与控制 作者&#xff1a;任晓旭 仇超 邓辉 戴子明 刘泽军 王晓飞 “人工智能技术与咨询” 发布 摘 要 边缘智能集网络、计算、存储和智能于一体&#xff0c;将智能推向网络边缘&#xff0c;为互联时代的低延迟关键计算开辟了道路。为进一步满足…

算法学习系列(四十):贡献法

目录 引言概念一、孤独的照片二、牛的基因学三、字串分值 引言 关于这个贡献法考的不是很多&#xff0c;主要题型是出现在需要枚举每一个组合这类题&#xff0c;出现的次数较多。没有固定的模板&#xff0c;就是一种思想&#xff0c;跟贪心一样&#xff0c;每个题都是不一样的…

Thymeleaf 基本使用

01、Thymeleaf 官网地址&#xff1a;Thymeleafhttps://www.thymeleaf.org/ 简介 Thymeleaf是一种服务器端Java模板引擎&#xff0c;用于将数据渲染为HTML、XML、JavaScript等格式&#xff0c;并在Web浏览器中呈现给用户。 具体来说&#xff0c;Thymeleaf充当着视图层的角色&…

C语言 ——注释

1.1 单行注释 - 语法&#xff1a;// 待注释的内容 - 位置&#xff1a;可放在代码后&#xff0c;称之为行尾注释&#xff1b; 也可放代码上一行&#xff0c;称作行上注释。 c // 这是单行注释文字 1.2 多行注释 - 语法&#xff1a;/* 待注释的内容 */ - 注意&#xff1a;多⾏…

考研C语言复习进阶(1)

目录 1. 数据类型介绍 1.1 类型的基本归类&#xff1a; 2. 整形在内存中的存储 2.1 原码、反码、补码 2.2 大小端介绍 3. 浮点型在内存中的存储 ​编辑 1. 数据类型介绍 前面我们已经学习了基本的内置类型&#xff1a; char //字符数据类型 short //短整型 int /…

前端之用HTML弄一个古诗词

将进酒 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>将进酒</title><h1><big>将进酒</big> 君不见黄河之水天上来</h1><table><tr><td ><img…

借助ChatGPT提高编程效率指南

PS: ChatGPT无限次数&#xff0c;无需魔法&#xff0c;登录即可使用,网页打开下面 一、借助ChatGPT提高编程效率指南 随着计算机技术的飞速发展&#xff0c;编程已经成为了现代社会中一个非常重要的技能。对于许多人来说&#xff0c;编程不仅是一项工作技能&#xff0c;而且是…

操作系统系列学习——操作系统之“树”

文章目录 前言操作系统之“树” 前言 一个本硕双非的小菜鸡&#xff0c;备战24年秋招&#xff0c;计划学习操作系统并完成6.0S81&#xff0c;加油&#xff01; 本文总结自B站【哈工大】操作系统 李治军&#xff08;全32讲&#xff09; 老师课程讲的非常好&#xff0c;感谢 【哈…

sqllab第五关通关笔记

知识点&#xff1a; 报错注入函数语法&#xff08;详见第二关笔记&#xff09;报错注入打印位数最多32位对于大于32位的数据最好使用截取函数进行控制&#xff1b;以保证输出完整mysql表中的重点数据库 information_schema &#xff08;mysql 5.0以上&#xff09; schemata …

每日一题 — 长度最小的子数组

LCR 008. 长度最小的子数组 - 力扣&#xff08;LeetCode&#xff09; 思路&#xff1a; 首先在暴力枚举的方法下&#xff0c;我们可以进行优化&#xff0c;让right不用从头开始&#xff0c;只需让left移动&#xff0c;进行判断。然后就是变成了同向双指针&#xff0c;也就是滑…

责任链模式-C#实现

责任链模式指的是——某个请求需要多个对象进行处理&#xff0c;从而避免请求的发送者和接收之间的耦合关系。 将这些对象连成一条链子&#xff0c;并沿着这条链子传递该请求&#xff0c;直到有对象处理它为止。 主要涉及两个角色&#xff1a; 抽象处理者角色&#xff08;Hand…

【测试开发学习历程】Linux用户管理+文件权限管理

目录 一、用户管理 &#xff08;一&#xff09;用户和用户组的基本概念 1.概念 2.设置原因 3.用户与用户组的关系 4.用户类型 &#xff08;二&#xff09;用户的创建、修改属性和删除用户 1.用户信息文件 2.创建用户 3.修改用户密码 4.修改用户信息 5.用户查询 6.…