【算法】归并排序

算法系列七:归并排序

一、归并排序的递归探寻

1.思路

2.搭建

2.1设计过掉不符情况(在最底层时)

2.2查验能实现基础排序(在最底层往上点时)

2.3跳转结果继续往上回搭

3.实质

4.实现

二、递归的调用栈

1.递归的执行过程

2.递归的函数栈帧

2.1递归函数的栈帧压弹

2.2合并有序数组函数的栈帧压弹

三、归并排序的复杂度

1.空间复杂度

2.时间复杂度


一、归并排序的递归探寻

1.思路

理想结果等于成分解断开子结果表达式的公式表示

快速排序:

整个数组有序 = 其中一个元素有序 + 其左断开数组有序 + 其右断开数组有序

归并排序

整个数组有序 = 左断开数组有序 + 右断开数组有序 + 两有序数组的有序合并


2.搭建

2.1设计过掉不符情况(在最底层时)

  • if(array == null) return, 数组没有元素不用排,下面是有元素
  • if(left == right) return,只有一个元素已经是有序了不用排,下面是多个元素
  • if(left > right) return,排不了不要排的,之后下面是符合一般情况的多个元素 

2.2查验能实现基础排序(在最底层往上点时)

在最底层往上点时,有序数组有序合并操作在最底层能实现两元素之间的比较然后进行排序的


2.3跳转结果继续往上回搭:

跳转有序数组结果继续往上有序合并维护回搭


3.实质

从底层的最小单个断开有序数组往上有序地合并成越来越大的断开有序数组直至合并完成一个整体的有序数组


4.实现

    public static void mergeSort(int[] array) {mergeSortFunc(array,0,array.length-1);}private static void mergeSortFunc(int[] array,int left,int right) {if(left >= right) return;int mid = (left+right) / 2;mergeSortFunc(array,left,mid);mergeSortFunc(array,mid+1,right);merge(array,left,right,mid);}private static void merge(int[] array, int left, int right, int mid) {int s1 = left;int s2 = mid+1;int[] tmpArr = new int[right-left+1];int k = 0;//证明两个区间 都同时有数据的while (s1 <= mid && s2 <= right) {if(array[s2] <= array[s1]) {tmpArr[k++] = array[s2++];}else {tmpArr[k++] = array[s1++];}}while (s1 <= mid) {tmpArr[k++] = array[s1++];}while (s2 <= right) {tmpArr[k++] = array[s2++];}//tmpArr 里面一定是这个区间内有序的数据了for (int i = 0; i < tmpArr.length; i++) {array[i+left] = tmpArr[i];}}

二、递归的调用栈

1.递归的执行过程

在函数递归中,调用的函数里面执行着再调用着随着形参深入的不断变化的函数自己

第一次调用函数的执行转去等着第二次调用函数的执行完,第二次调用函数的执行也卡着转去等第三次调用函数的执行完,一层层形参变化着重复地执行调用而都没有往下去return直到最后调用函数传的形参变化到符合return的条件不再继续往下调用了return出结果开始往回地一层层促进上一层没有执行完到执行完return,即开始往回地执行往回地归


2.递归的函数栈帧

所有任意一个函数的调用都会独立开辟新的函数栈帧(里面存放局部变量、函数形参、返回地址、寄存器值)压入调用栈中,在函数执行到return语句结束后才弹出栈

递归调用时,函数栈帧从先往后地一个个独立地压入栈中,往下递归直到形参条件变到return由最新函数调用的栈帧开始往上弹栈帧出调用栈,在开始往上弹出栈帧开始有执行完往回层往下执行时,方法里面有可能写的是继续又去执行调用当前层形参条件下的再来一波函数递归调用(形参变化可能就去设置成不同的了就会是左右不一样的分支),即是二叉即二叉树的情况:

  • 每一层不仅要左边往下全执行到底层然后开始往上全执行到它那层的左边结束,接着又要执行右边的往下执行到底层然后再往上全执行完到它的右层结束最后它这个节点对应的这层函数才也执行完它的栈帧也弹出调用栈,二叉树从大到小所有的结构都是左边往下全执行完往上回来、右边往下全执行完往上回来、接着它这个节点的栈帧也往上弹返回执行完 

2.1递归函数的栈帧压弹

在归并排序的二叉树递归调用过程中:

  1. 每次累计着往下调用到底层时,此时的调用栈所占的空间是最大的、深度在二叉树的最底层,调用栈的空间计算为调用栈里栈帧的个数×每个栈帧的内存大小,在每个函数的栈帧中,函数里面那些函数的调用信息并非循环出现的常量个数的局部变量空间和都可算成常数的大小,所以在归并排序这里,调用栈的最大空间为在调用栈里栈帧个数最多的时候:树的高度log(n)*每个函数栈帧的内存大小是常数,即log(n)*常数函数调用执行完最底层后就开始有往上返回了,往上弹出最新最顶的栈帧
  2. 然后执行完返回到上层时又回到当前层条件下的且新的形参变化模式的再往下递归,又会去压栈到最底层,此时调用栈的空间又达到最大的log(n)
  3. 当它右边往下的也全执行完又往上返回到当前层时,就开始继续往下接着执行就开始有去调用合并有序数组的函数了

2.2合并有序数组函数的栈帧压弹

执行调用合并有序数列的函数时,调用栈又会压入合并有序数组函数的栈帧,里面存放有开辟的当前层数组元素个数大小的数组(非常量级的,要算的),此时总的占用空间为调用栈的空间log(n-...)+n(-...),因为合并有序数组函数的栈帧每次都是处在栈顶压入的函数里面并没有再调用函数的在它之上再压栈,所以它每次在栈顶进来压栈完就紧接着弹出栈的


三、归并排序的复杂度

1.空间复杂度

空间复杂度计算的是整个执行所有时刻中出现的最大瞬时占用空间

从下层往上层的返回的过程中,递归函数的调用栈空间变小着、合并有序数组的函数栈帧在变大着(里面的数组越来越大的):

  • 最底层时占用的空间为递归函数的调用栈空间log(n)+合并有序数组的函数栈帧0,即log(n)+0=log(n)
  • 当到达最上层第一层时,递归函数的调用栈空间是1,而合并有序数组的函数栈帧空间是n,此时的总空间大小是n,相比于最底层的log(n)及从下往上的过程中log(n)的递减、n的递增的总空间,此时的n是整个执行所有时刻中出现的最大瞬时占用的空间,所以归并排序的空间复杂度是O(n)


2.时间复杂度

时间复杂度即算整个递归调用执行过程的时间和,我们可以不用按着递归搜索的过程去时时累计总的算,直接站在总二叉树的角度一层一层地算所有时间的和就行了一层层里面每一个树节点及下的全执行完对应着该调用函数的全执行完,因为递归调用语句mergeSortFunc(array,left,mid)都是且已转成里面的函数节点内容来算了(调用中的去执行调用部分是常量级的已不算),且if(left >= right) return、int mid = (left+right) / 2也都是常量级的执行时间不算对应到总的时间就是计算所有函数节点里的merge(array,left,right,mid)合并有序数组的时间和每一层所有函数节点的合并有序数组时间和都为n(除了最后一层的函数节点进去就直接判断为return没执行有序数组合并),一共有log(n)层,所以时间复杂度为O(n*log(n)) 

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

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

相关文章

线束线缆从二维设计到虚拟验证全流程解决方案

一、传统设计中的痛点 线缆的开发设计是横跨多专业多学科的庞大工程&#xff0c;通常会划分为几大阶段逐次推进&#xff0c;由于每个阶段的工作任务不同&#xff0c;所以在不同设计阶段使用的工具也完全不同&#xff0c;由此导致整个设计流程中工程师常常要跨平台协作&#xf…

【智驾中的大模型 -1】自动驾驶场景中的大模型

1. 前言 我们知道&#xff0c;大模型现在很火爆&#xff0c;尤其是 deepseek 风靡全球后&#xff0c;大模型毫无疑问成为为中国新质生产力的代表。百度创始人李彦宏也说&#xff1a;“2025 年可能会成为 AI 智能体爆发的元年”。 随着科技的飞速发展&#xff0c;大模型的影响…

个人博客系统后端 - 注册登录功能实现指南

一、功能概述 个人博客系统的注册登录功能包括&#xff1a; 用户注册&#xff1a;新用户可以通过提供用户名、密码、邮箱等信息创建账号用户登录&#xff1a;已注册用户可以通过用户名和密码进行身份验证&#xff0c;获取JWT令牌身份验证&#xff1a;使用JWT令牌访问需要认证…

投行交易与风控系统的消费侧幂等架构设计与实战

1.背景和痛点 1.1 资金操作敏感性场景 核心需求&#xff1a; 交易唯一性&#xff1a;资金类操作必须保证全局唯一执行计算原子性&#xff1a;风控指标计算需具备事务性特征审计追溯&#xff1a;所有操作需保留完整幂等轨迹 1.2 业务损失统计 二、技术挑战与架构设计 2.1 分…

odoo-046 视图显示的 name 数据库中存储的不一样

文章目录 一、问题由来二、排查经过1. 问 deepseek2. 验证3. 新问题 三、 总结四、补充&#xff08;翻译模型 ir.translation 中 src 和 value 字段详解&#xff09; 一、问题由来 客户有多个公司&#xff0c;使用多个数据库。他们有时需要同步不同数据库之间的数据的需求。在…

充电宝项目:规则引擎Drools学习

文章目录 规则引擎 Drools1 问题2 规则引擎概述2.1 规则引擎2.2 使用规则引擎的优势2.3 规则引擎应用场景2.4 Drools介绍 3 Drools入门案例3.1 创建springboot项目 引入依赖3.2 添加Drools配置类3.4 创建实体类Order3.5 orderScore.drl3.6 编写测试类 4 Drools基础语法4.1 规则…

HTML、CSS 和 JavaScript 常见用法及使用规范

一、HTML 深度剖析 1. 文档类型声明 HTML 文档开头的 <!DOCTYPE html> 声明告知浏览器当前文档使用的是 HTML5 标准。它是文档的重要元信息&#xff0c;能确保浏览器以标准模式渲染页面&#xff0c;避免怪异模式下的兼容性问题。 2. 元数据标签 <meta> 标签&am…

基于CNN+ViT的蔬果图像分类实验

本文只是做一个简单融合的实验&#xff0c;没有任何新颖&#xff0c;大家看看就行了。 1.数据集 本文所采用的数据集为Fruit-360 果蔬图像数据集&#xff0c;该数据集由 Horea Mureșan 等人整理并发布于 GitHub&#xff08;项目地址&#xff1a;Horea94/Fruit-Images-Datase…

Ubuntu24.04安装libgl1-mesa-glx 报错,软件包缺失

在 Ubuntu 24.04 系统中&#xff0c;您遇到的 libgl1-mesa-glx 软件包缺失问题可能是由于该包在最新的 Ubuntu 版本中被重命名为 libglx-mesa0。以下是针对该问题的详细解决方案&#xff1a; 1. 问题原因分析 包名称变更&#xff1a;在 Ubuntu 24.04 中&#xff0c;libgl1-me…

webpack vite

​ 1、webpack webpack打包工具&#xff08;重点在于配置和使用&#xff0c;原理并不高优。只在开发环境应用&#xff0c;不在线上环境运行&#xff09;&#xff0c;压缩整合代码&#xff0c;让网页加载更快。 前端代码为什么要进行构建和打包&#xff1f; 体积更好&#x…

如何在爬虫中合理使用海外代理?在爬虫中合理使用海外ip

我们都知道&#xff0c;爬虫工作就是在各类网页中游走&#xff0c;快速而高效地采集数据。然而如果目标网站分布在多个国家或者存在区域性限制&#xff0c;那靠普通的网络访问可能会带来诸多阻碍。而这时&#xff0c;“海外代理”俨然成了爬虫工程师们的得力帮手&#xff01; …

数据仓库分层存储设计:平衡存储成本与查询效率

数据仓库分层存储不仅是一个技术问题,更是一种艺术:如何在有限的资源下,让数据既能快速响应查询,又能以最低的成本存储? 目录 一、什么是数据仓库分层存储? 二、分层存储的体系架构 1. 数据源层(ODS,Operational Data Store) 2. 数据仓库层(DW,Data Warehouse)…

YOLO学习笔记 | 基于YOLOv8的植物病害检测系统

以下是基于YOLOv8的植物病害检测系统完整技术文档,包含原理分析、数学公式推导及代码实现框架。 基于YOLOv8的智能植物病害检测系统研究 摘要 针对传统植物病害检测方法存在的效率低、泛化性差等问题,本研究提出一种基于改进YOLOv8算法的智能检测系统。通过设计轻量化特征提…

高级语言调用C接口(二)回调函数(4)Python

前面2篇分别说了java和c#调用C接口&#xff0c;参数为回调函数&#xff0c;回调函数中参数是结构体指针。 接下来说下python的调用方法。 from ctypes import * import sysclass stPayResult(Structure):_pack_ 4 # 根据实际C结构体的对齐方式设置&#xff08;常见值为1,4,…

springboot启动动态定时任务

1.自定义定时任务线程池 package com.x.devicetcpserver.global.tcp.tcpscheduler;import org.springframework.boot.context.properties.EnableConfigurationProperties; import org.springframework.context.annotation.Bean; import org.springframework.context.annotatio…

pytorch框架认识--手写数字识别

手写数字是机器学习中非常经典的案例&#xff0c;本文将通过pytorch框架&#xff0c;利用神经网络来实现手写数字识别 pytorch中提供了手写数字的数据集&#xff0c;我们可以直接从pytorch中下载 MNIST中包含70000张手写数字图像&#xff1a;60000张用于训练&#xff0c;10000…

WPF 使用依赖注入后关闭窗口程序不结束

原因是在ViewModel中在构造函数中注入了Window 对象&#xff0c;即使没有使用&#xff0c;主窗口关闭程序不会退出&#xff0c;即使 ViewModel 是 AddTransient 注入的。 解决方法&#xff1a;不使用构造函数注入Window&#xff0c;通过GetService获取Window 通过注入对象调用…

用户管理(添加和删除,查询信息,切换用户,查看登录用户,用户组,配置文件)

目录 添加和删除用户 查询用户信息 切换用户 查看当前的操作用户是谁 查看首次登录的用户是谁 用户组&#xff08;对属于同个角色的用户统一管理&#xff09; 新增组 删除组 添加用户的同时&#xff0c;指定组 修改用户的组 组的配置文件&#xff08;/etc/group&…

PyTorch学习-小土堆教程

网络搭建torch.nn.Module 卷积操作 torch.nn.functional.conv2d(input, weight, biasNone, stride1, padding0, dilation1, groups1) 神经网络-卷积层

MVCC详细介绍及面试题

目录 1.什么是mvcc&#xff1f; 2.问题引入 3. MVCC实现原理&#xff1f; 3.1 隐藏字段 3.2 undo log 日志 3.2.1 undo log版本链 3.3 readview 3.3.1 当前读 ​编辑 3.3.2 快照读 3.3.3 ReadView中4个核心字段 3.3.4 版本数据链访问的规则&#xff08;了解&#x…