三大排序算法

2023-11-08

目录

大O和推导过程

冒泡排序

冒泡排序的思路:

冒泡排序的实现

 冒泡排序的效率  O(N²)

选择排序

选择排序的思路

选择排序实现

选择排序的效率  O(N²)

插入排序

插入排序的思路

插入排序的实现

插入排序的效率  O(N²)


大O和推导过程

公司规模的概述

公司可以按照规模分为:小型/中型/大型企业

在不说明具体员工人数或者占地面积的情况下,我们可以通过这样大概的概念来描述企业的规模

大O表示法

  • 在算法的描述中,我们也可以通过类似的快捷方式来描述计算机的算法效率
  • 在计算机中,这种粗略的变量被称为 大O表示法
  • 在数据各项发生变化时,算法的效率会跟着发生改变
  • 所以我们通常使用一种 算法的速度会如何跟随着数据量变化的、 

推导大O表示法的方式

  • 用常量1取代运行时间中所有的加法常量
  • 在修改后运行次数函数中。只保留最高的阶级
  • 如果最高存在且不为1,则去除与这个项相乘的常量

冒泡排序

冒泡排序算法相对其他排序运行效率较低, 但是在概念上它是排序算法中最简单的.

冒泡排序的思路:

  • 对未排序的各元素从头到尾依次比较相邻的两个元素大小关系
  • 如果左边的队员高, 则两队员交换位置
  • 向右移动一个位置, 比较下面两个队员
  • 当走到最右端时, 最高的队员一定被放在了最右边
  • 按照这个思路, 从最左端重新开始, 这次走到倒数第二个位置的队员即可.
  • 依次类推, 就可以将数据排序完成

思路再分析:

  • 第一次找出最高人放在最后, 我们需要两个两个数据项进行比较, 那么这个应该是一个循环操作.
  • 第二次将次高的人找到放在倒数第二个位置, 也是两个比较, 只是不要和最后一个比较(少了一次), 但是前面的两个两个比较也是一个循环操作.
  • 第三次...第四次...
  • 有发现规律吗? 这应该是一个循环中嵌套循环, 并且被嵌套的循环次数越来越少的.

冒泡排序的实现

    <script>
        let arr = [3, 8, 7, 4, 11, 62, 5]
        /* 
        第一次 j=length-1 比较到倒数读一个位置
        第二次 j=length-2 比较到倒数读二个位置
         */
        function bubblesSort(arr) {
            //获取数组长度
            let len = arr.length
            /*    第一次进来,i=0,比较0和1的位置的两个数据, 如果0的位置,大于1的位置交换数据
                 最后一次进来,i=length-2, 比较length-2,和length-1的数据
            */
            for (let j = len - 1; j >= 0; j--) {
                for (let i = 0; i < j; i++) {
                    if (arr[i] > arr[i + 1]) {
                        //交互数据
                        let temp = arr[i]
                        arr[i] = arr[i + 1]
                        arr[i + 1] = temp
                    }
                }

            }
            return arr
        }
        console.log(bubblesSort(arr));
    </script>

代码解析:

  •  获取数组的长度.
  •  我们现在要写的外层循环, 外层循环应该让i依次减少, 因此我们这里使用了反向的遍历.
  • 内层循环, 内层循环我们使用 j < i. 因为上面的i在不断减小, 这样就可以控制内层循环的次数.
  • : 比较两个数据项的大小, 如果前面的大, 那么就进行交换.

 

 冒泡排序的效率  O(N²)

  • 冒泡排序的比较次数:
    • 如果按照上面的例子来说, 一共有7个数字, 那么每次循环时进行了几次的比较呢?
    • 第一次循环6次比较, 第二次5次比较, 第三次4次比较....直到最后一趟进行了一次比较.
    • 对于7个数据项比较次数: 6 + 5 + 4 + 3 + 2 + 1
    • 对于N个数据项呢? (N - 1) + (N - 2) + (N - 3) + ... + 1 = N * (N - 1) / 2
  • 大O表示法:
    • 大O表示法是描述性能和复杂度的一种表示方法.
    • 推导大O表示法通常我们会使用如下规则:
      • 用常量1取代运行时间中的所有加法常量
      • 在修改后的运行次数函数中, 只保留最高阶项
      • 如果最高阶项存在并且不是1, 则去除与这个项相乘的常数.
  • 通过大O表示法推到过程, 我们来推到一下冒泡排序的大O形式.
    • N * (N - 1) / 2 = N²/2 - N/2,根据规则2, 只保留最高阶项, 编程N² / 2
    • N² / 2, 根据规则3, 去除常量, 编程N²
    • 因此冒泡排序的大O表示法为O(N²)
  • 冒泡排序的交换次数:
    • 如果有两次比较才需要交换一次(不可能每次比较都交换一次.), 那么交换次数为N² / 4
    • 由于常量不算在大O表示法中, 因此, 我们可以认为交换次数的大O表示也是O(N²)

选择排序

选择排序的思路

  • 选择排序的思路:

    • 选定第一个索引位置,然后和后面元素依次比较
    • 如果后面的队员, 小于第一个索引位置的队员, 则交换位置
    • 经过一轮的比较后, 可以确定第一个位置是最小的
    • 然后使用同样的方法把剩下的元素逐个比较即可
    • 可以看出选择排序,第一轮会选出最小值,第二轮会选出第二小的值,直到最后

思路再分析:

  • 选择排序第一次将第0位置的人取出, 和后面的人(1, 2, 3...)依次比较, 如果后面的人更小, 那么就交换.
  • 这样经过一轮之后, 第一个肯定是最小的人.
  • 第二次将第1位置的人取出, 和后面的人(2, 3, 4...)依次比较, 如果后面的人更小, 那么就交换.
  • 这样经过第二轮后, 第二个肯定是次小的人.
  • 第三轮...第四轮...直到最后就可以排好序了
  • 外层循环依次取出0-1-2...N-2位置的人作为index(N-1不需要取了, 因为只剩它一个了肯定是排好序的)
  • 内层循环从index+1开始比较, 直到最后一个.

选择排序实现

    <script>
        let arr = [3, 8, 7, 4, 11, 62, 5]
        function slectionSort(arr) {
            //获取数长度
            let len = arr.length
            // 外层循环:从0开始获取数据
            for (let j = 0; j < arr.length - 1; j++) {
                let min = j
                // 内层循环:从 i+1位置开始,和后面的数据进行比较
                for (let i = min + 1; i < len; i++) {
                    //如果i的为位置数据大于j的位置数据,那么记录最小位置 
                    if (arr[min] > arr[i]) {
                        min = i
                    }
                    let temp = arr[min]
                    arr[min] = arr[j]
                    arr[j] = temp

                }
            }

            return arr
        }
        console.log(slectionSort(arr));
    </script>

代码解析:

  • 依然获取数组的长度.
  • 外层循环, , 需要从外层循环的第0个位置开始, 依次遍历到length - 2的位置.
  •  先定义一个min, 用于记录最小的位置, 内层循环, 内层循环是从i+1位置开始的数据项, 和i位置的数据项依次比较, 直到length-1的数据项.
  •  如果比较的位置i的数据项, 大于后面某一个数据项, 那么记录最小位置的数据.
  • : 将min位置的数据, 那么i位置的数据交换, 那么i位置就是正确的数据了.

选择排序的效率  O(N²)

  • 选择排序的比较次数:
    • 选择排序和冒泡排序的比较次数都是N*(N-1)/2, 也就是O(N²).
  • 选择排序的交换次数:
    • 选择排序的交换次数只有N-1次, 用大O表示法就是O(N).
    • 所以选择排序通常认为在执行效率上是高于冒泡排序的.

插入排序

插入排序的思路

  • 局部有序:

    • 插入排序思想的核心是局部有序. 
    • 比如在一个队列中的人, 我们选择其中一个作为标记的队员. 这个被标记的队员左边的所有队员已经是局部有序的.
    • 这意味着, 有一部门人是按顺序排列好的. 有一部分还没有顺序.
  • 插入排序的思路:

    • 从第一个元素开始,该元素可以认为已经被排序
    • 取出下一个元素,在已经排序的元素序列中从后向前扫描
    • 如果该元素(已排序)大于新元素,将该元素移到下一位置
    • 重复上一个步骤,直到找到已排序的元素小于或者等于新元素的位置
    • 将新元素插入到该位置后, 重复上面的步骤.

思路再分析:

  • 插入排序应该从下标值1开始(因为0位置默认可以被认为是有序的)
  • 从1位置开始取出元素, 并且判断该元素的大小和0位置进行比较, 如果1位置元素小于0位置元素, 那么交换, 否则不交换.
  • 上面步骤执行完成后, 0 - 1位置已经排序好.
  • 取出2位置的元素, 和1位置进行比较:
    • 如果2位置元素大于1位置元素, 说明2位置不需要任何动作. 0 - 1 - 2已经排序好.
    • 如果2位置元素小于1位置元素, 那么将1移动到2的位置, 并且2继续和0进行比较.
    • 如果2位置元素大于0位置的元素, 那么将2位置放置在1的位置, 排序完成. 0 - 1 - 2搞定.
    • 如果2位置元素小于1位置的元素, 那么将0位置的元素移动到1位置, 并且将2位置的元素放在0位置, 0 - 1 - 2搞定.
  • 按照上面的步骤, 依次找到最后一个元素, 整个数组排序完成.0

插入排序的实现

    <script>
        let arr = [3, 8, 7, 4, 11, 62, 5]
        function insertSort(arr) {
            //获取数长度
            let len = arr.length
            // 外层循环从第一个位置开始获取数据,向前面局部有序的进行插入
            for (let i = 1; i < len; i++) {
                //内层循环:获取i位置的元素,和前面的一次进行比较
                let temp = arr[i]
                let j = i
                while (arr[j - 1] > temp && j > 0) {
                    arr[j] = arr[j - 1]
                    j--
                }
                arr[j] = temp
            }


            return arr
        }
        console.log(slectionSort(arr));
    </script>

 

  •  · 获取数组的长度.
  •  外层循环, 从1位置开始, 因为0位置可以默认看成是有序的了.
  •  记录选出的i位置的元素, 保存在变量temp中. i默认等于j
  • 内层循环
    • 内层循环的判断j - 1位置的元素和temp比较, 并且j > 0.
    • 那么就将j-1位置的元素放在j位置.
    • j位置向前移.
  • :将目前选出的j位置放置temp元素.

插入排序的效率  O(N²)

  • 插入排序的比较次数:
    • 第一趟时, 需要的最多次数是1, 第二趟最多次数是2, 依次类推, 最后一趟是N-1次.
    • 因此是1 + 2 + 3 + ... + N - 1 = N * (N - 1) / 2.
    • 然而每趟发现插入点之前, 平均只有全体数据项的一半需要进行比较.
    • 我们可以除以2得到 N * (N - 1) / 4. 所以相对于选择排序, 其他比较次数是少了一半的.
  • 插入排序的复制次数:
    • 第一趟时, 需要的最多复制次数是1, 第二趟最多次数是2, 依次类推, 最后一趟是N-1次.
    • 因此是1 + 2 + 3 + ... + N - 1 = N * (N - 1) / 2.
  • 对于基本有序的情况
    • 对于已经有序或基本有序的数据来说, 插入排序要好很多.
    • 当数据有序的时候, while循环的条件总是为假, 所以它变成了外层循环中的一个简单语句, 执行N-1次.
    • 在这种情况下, 算法运行至需要N(N)的时间, 效率相对来说会更高.
    •  我们的比较次数是选择排序的一半, 所以这个算法的效率是高于选择排序的.。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

三大排序算法 的相关文章

  • vue2中Cascader 级联选择器限制选择个数和回显问题

    文章目录 1 组件默认数据绑定 2 指定数据绑定 3 watch监听v model绑定的数组 控制选中个数 4 前后端数据转换 实现回显 1 接口初始数据回显 2 重新选择级联选择器后 如何将选择的数据转换成后端需要的数据 3 最后提交数据
  • 搜索二叉树(BSTree)

    一 搜索二叉树的概念 二叉搜索树又称为做二叉排序树 二叉查找树 其要么是一棵空树 要么是一个满足以下性质的二叉树 若它的左子树不空 则左子树上所有结点的关键字均小于根结点关键字 若它的右子树不空 则右子树上所有结点的关键字均大于根结点关键字
  • 【网络安全】web漏洞-xml外部实体注入(XXE)

    web漏洞 xml外部实体注入 XXE 目录 web漏洞 xml外部实体注入 XXE 概念 危害 检测方法 利用方法 漏洞利用 xxe lab 有回显情况 无回显情况 pikachu靶场
  • 如何给 unplugin-vue-components/vite 写一个简单的 resolver

    大部分工作 unplugin vue components 都已经处理好了 我们只需要接收组件名来判断是否是自己的组件 然后处理对应的导入逻辑 一共 3 个字段 as 重命名类似 import componentNameReName fro
  • Web 安全漏洞之 OS 命令注入

    什么是 OS 命令注入 上周我们分享了一篇 Web 安全漏洞之 SQL 注入 其原理简单来说就是因为 SQL 是一种结构化字符串语言 攻击者利用可以随意构造语句的漏洞构造了开发者意料之外的语句 而今天要讲的 OS 命令注入其实原理和 SQL
  • 38条Web测试经验分享

    1 页面链接检查 每一个链接是否都有对应的页面 并且页面之间切换正确 可以使用一些工具 如LinkBotPro File AIDCS HTML Link Validater Xenu等工具 LinkBotPro不支持中文 中文字符显示为乱码
  • 前端必备的 web 安全知识手记

    前言 安全这种东西就是不发生则已 一发生则惊人 作为前端 平时对这方面的知识没啥研究 最近了解了下 特此沉淀 文章内容包括以下几个典型的 web 安全知识点 XSS CSRF 点击劫持 SQL 注入和上传问题等 下文以小王代指攻击者 话不多
  • WEB前端常见受攻击方式及解决办法总结

    一个网址建立后 如果不注意安全问题 就很容易被人攻击 下面讨论一下集中漏洞情况和放置攻击的方法 一 SQL注入 所谓的SQL注入 就是通过把SQL命令插入到web表单提交或输入域名或页面请求的查询字符串 最终达到欺骗服务器执行恶意的SQL命
  • 每天10个前端小知识 <Day 7>

    前端面试基础知识题 1 什么是尾调用优化和尾递归 尾调用的概念非常简单 一句话就能说清楚 就是指某个函数的最后一步是调用另一个函数 function f x return g x 上面代码中 函数f的最后一步是调用函数g 这就叫尾调用 尾调
  • 基于java的饮食分享平台系统设计与实现

    基于java的饮食分享平台系统设计与实现 I 引言 A 研究背景和动机 近年来 随着人们生活水平的提高和健康意识的增强 饮食健康已经成为越来越多人的关注焦点 因此 一个方便快捷的饮食分享平台就显得尤为重要 基于Java的饮食分享平台系统设计
  • 软件测试|web自动化测试神器playwright教程(三十八)

    简介 在我们使用selenium时 我们可以获取元素的属性 元素的文本值 以及输入框的内容等 作为比selenium更为强大的web自动化测试神器 playwright也可以实现对元素属性 文本值和输入框内容的抓取 并且实现比seleniu
  • Vue3 和Vue2的区别,以及钩子函数的使用

    Vue js 3 和 Vue js 2 是两个主要版本的流行前端框架 它们之间有很多区别 包括性能优化 新特性和改进的API等 以下是一些Vue 3与Vue 2之间的主要区别 以及一些示例代码来说明这些差异 1 性能优化 响应式系统 Vue
  • 低代码配置-组件列表设计

    过滤字段功能 配置了api 启用 输出配置 filter type Array default gt
  • 「优选算法刷题」:快乐数

    一 题目 编写一个算法来判断一个数 n 是不是快乐数 快乐数 定义为 对于一个正整数 每一次将该数替换为它每个位置上的数字的平方和 然后重复这个过程直到这个数变为 1 也可能是 无限循环 但始终变不到 1 如果这个过程 结果为 1 那么这个
  • 考虑光伏出力利用率的电动汽车充电站能量调度策略研究(Matlab代码实现)

    欢迎来到本博客 博主优势 博客内容尽量做到思维缜密 逻辑清晰 为了方便读者 座右铭 行百里者 半于九十 本文目录如下 目录 1 概述 2 运行结果 3 参考文献 4 Matlab代码 数据
  • Web自动化测试 —— cookie复用

    一 cookie简介 cookie是一些数据 存储于用户电脑的文本文件中 当web服务器想浏览器发送web页面时 在链接关闭后 服务端不会记录用户信息 二 为什么要使用Cookie自动化登录 复用浏览器仍然在每次用例开始都需要人为介入 若用
  • 每天10个前端小知识 <Day 14>

    前端面试基础知识题 1 CSSOM树和DOM树是同时解析的吗 浏览器会下载HTML解析页面生成DOM树 遇到CSS标签就开始解析CSS 这个过程不会阻塞 但是如果遇到了JS脚本 此时假如CSSOM还没有构建完 需要等待CSSOM构建完 再去
  • 【js学习之路】遍历数组api之 `filter `和 `map`的区别

    一 前言 数组是我们在项目中经常使用的数据类型 今天我们主要简述作用于遍历数组的api filter 和 map 的区别 二 filter和map的共同点 首先 我们主要阐述一下 filter 和 map 的共同点 api的参数都是回调函数
  • 如何在 Python 脚本中使用 Google OAuth2

    在使用 Python 脚本将视频上传到 YouTube 频道时 若希望将视频上传到第二个频道 需要解决 OAuth2 授权的问题 解决方案 创建新的 Google Cloud 项目 from google oauth2 import ser
  • 【前端】canvas图片加文字

    注释标记了操作步骤 import React Component createRef from react class CertifyImgRender extends Component bgRef createRef

随机推荐

  • mybatis-generator自动生成的类中含有XXXwithBLOBs,去掉的方法

    当数据库中的字段有text类型时 mybatis会为这种类型单独创建一个类来映射这两个字段 生成的主要po类中是没有这两个字段的 自动生成的xxxWithBLOBs类会继承生成的主要po类 public class ProductWithB
  • 束缚游戏 html,束缚游戏

    束缚 描述的是一个锁链束缚的眼镜男 在游戏中会遇到各种障碍 探索一个普通居家男人的心灵利用铁球来通过这些障碍的横向平台解谜游戏 游戏简介 束缚 是一款卷轴平台益智游戏 主角是一个被锁链束缚的眼镜男 他可以利用铁球通过各种障碍 游戏的宗旨是探
  • linux与freertos程序兼容,从freeRTOS运行应用程序

    FreeRTOS 以及大多数RTOS 不像通用操作系统 GPOS 那样工作 它们通常不是为了动态加载和执行任意用户提供的应用程序而设计的 在大多数情况下 您使用RTOS是因为您需要硬实时响应 并且执行第三方代码可能会对此造成影响 大多数RT
  • vivado AXI_interconnector ID信号的一些总结

    很久没有写东西了 最近尽力了很多生活上的事情 最终也算圆满结局 言归正传 主要是以AXI4为背景介绍4组ID信号 以及其计算方式 对照vivado PG059以及PG247 见解都是基于个人所学 会有偏差还望谅解 AXI4中去掉了WID 所
  • 基于iframe的HTTP长连接实现

    关于什么是http长连接我不废吐沫了 有专业的解释 http www ibm com developerworks cn web wa lo comet 你可以去看看我们介绍一下在struts下的实现首先写一个test jsp 写一些片段
  • 【100天精通python】Day30:使用python操作数据库_数据库基础入门

    专栏导读 专栏订阅地址 https blog csdn net qq 35831906 category 12375510 html 1 数据库基础知识介绍 1 1 什么是数据库 数据库是一个结构化存储和组织数据的集合 它可以被有效地访问
  • android 网络自动同步时间慢问题

    问题描述 今天测试提了一个网络同步时间慢的bug 网络同步时间原理参考 https blog csdn net yin1031468524 article details 65447849 核心代码在NetworkTimeUpdateSer
  • 【遗传算法】【处理图像类问题】

    文章目录 一 前言 二 问题描述 三 算法介绍 四 其他知识点 Reference 一 前言 近期感兴趣的算法 以前没这么好奇过一个算法 时间没想象的焦虑 认真做一些事情 算法入门篇 二 问题描述 从前 一群扇贝在海岸边悠哉游哉地生活着 它
  • ThinkPHP3.2自带的七牛云配置使用

    利用七牛云私有空间存储文件 第一步 注册七牛云 创建空间 将空间设为私有 需要记下的东西 accessKey secrectKey domain bucket 第二步配置ThinkPHP 在config php添加 UPLOAD SITEI
  • STM32 FreeRTOS 内存问题

    1 STM32L151C8T6 内存 64Kb 的Flash 代码就是烧录在这里面的 16Kb 的RAM 程序跑起来之后的内存 相当于我们高考时发的草稿纸 直接影响程序的运行速度 可以用STM32 CubeMx 软件直接下载数据手册data
  • Calendar 中getActualMaximumd 功能

    String str new SimpleDateFormat yyyy MM dd HH mm ss SSS format new Date Calendar calendar Calendar getInstance Locale CH
  • 使用pytorch训练DCGAN----贰(代码解析)

    使用pytorch训练DCGAN 代码解析 上一篇 使用pytorch训练DCGAN 壹 这里我使用的代码时pytorch官方提供的源码 然后根据DCGAN原论文分板块分析 板块一 导包 from future import print f
  • javaWeb图书管理系统

    javaWeb图书管理系统 1 项目简单介绍 a 项目用到的技术 IDE Intellij IDEA 语言 java html ajax js 数据库 Mysql 数据库可视化 navicat web服务器 Tomcat 框架 mybati
  • C++使用当multiset插入相同的数时 新插入的数在已有数的左边还是右边呢

    答案 在multiset中 元素按照特定的顺序进行排序并存储 当向multiset插入相同的数时 新插入的数将被放置在已有数的右边 multiset允许存储重复的元素 并且保持了元素的顺序 因此 如果已经存在相同的数 新插入的数将被放置在它
  • c++在线编辑器

    c 在线编辑器 Compiler Explorer Coliru Ideone 乱糟糟的不推荐 C Shell CodingGround 可用来美化代码 慢的很 Judge0 IDE Compiler Explorer https godb
  • P1088 [NOIP2004 普及组] 火星人(全排列)

    题目链接 火星人 思路分析 分析题目题意得到 这个题目关于全排列的问题 题目输入N M分别代表对1 N进行全排列 火星人给出的大数就是1 N全排列的一种情况 对全排列的所有情况进行编号 例如下图 例如上图N 3 假设M 2 给出一种全排列序
  • ffmpeg推流收流 1920*1080视频 花屏

    自己用ffmpeg推流 然后再收流 小分辨率没有问题 当分辨率为1920 1080时 出现花屏现象 尤其是码率高时 现象更加明显 尝试各种办法 最后用下面的办法解决 在ffmpeg源码udp c中 define UDP MAX PKT SI
  • Ubuntu22.04配置静态IP-网关-DNS

    要在Ubuntu系统中配置网络 可以通过以下步骤进行操作 1 打开终端 可以使用 Ctrl Alt T 快捷键打开终端 或者从应用程序菜单中找到 终端 2 检查网络接口 输入以下命令检查当前系统中的网络接口列表 ifconfig a 接口列
  • 贝叶斯网络python实战(以泰坦尼克号数据集为例,pgmpy库)

    文章目录 贝叶斯网络简介 贝叶斯推断思路 贝叶斯网络 贝叶斯网络的实现 应用步骤 泰坦尼克数据集背景介绍 模型结构搭建 模型参数构建 贝叶斯估计器 推理 自动设计网络结构 gt 使用结构学习方法 模型保存 先验 练手数据集 Binary C
  • 三大排序算法

    目录 大O和推导过程 冒泡排序 冒泡排序的思路 冒泡排序的实现 冒泡排序的效率 O N 选择排序 选择排序的思路 选择排序实现 选择排序的效率 O N 插入排序 插入排序的思路 插入排序的实现 插入排序的效率 O N 大O和推导过程 公司规