【算法】归并排序

【算法】归并排序

算法系列七:归并排序

一、归并排序的递归探寻

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)) 

Read more

Java霸主未逝:不可撼动的生态与新特性的革命潜力

Java霸主未逝:不可撼动的生态与新特性的革命潜力

🧑 博主简介:ZEEKLOG博客专家,历代文学网(PC端可以访问:https://literature.sinhy.com/#/?__c=1000,移动端可微信小程序搜索“历代文学”)总架构师,15年工作经验,精通Java编程,高并发设计,Springboot和微服务,熟悉Linux,ESXI虚拟化以及云原生Docker和K8s,热衷于探索科技的边界,并将理论知识转化为实际应用。保持对新技术的好奇心,乐于分享所学,希望通过我的实践经历和见解,启发他人的创新思维。在这里,我希望能与志同道合的朋友交流探讨,共同进步,一起在技术的世界里不断学习成长。 技术合作请加本人wx(注明来自ZEEKLOG):foreast_sea Java霸主未逝:不可撼动的生态与新特性的革命潜力 引言:在编程语言的巨变时代重新审视Java 在技术飞速演进的时代,编程语言的世界仿佛一片汹涌的海洋,每天都有新的语言和框架涌现,声称要颠覆现有秩序。从Python在数据科学和人工智能领域的崛起,到Go语言在并发处理和高性能网络服务中的优异表现,再到Rust在系统编程和安全关键型应用中的强势进攻,似乎每一种语

By Ne0inhk

AI大模型实用(三)Java快速实现智能体整理(Springboot+LangChain4j)

目录 1.1 简介 1.2 示例 步骤一: 添加pom 步骤二:配置 步骤三:流式输出 步骤四: 正常输出 步骤五: 【类似函数调用】AI Service接口 1.3 调试问题 问题1: ClassNotFoundException: dev.langchain4j.exception.IllegalConfigurationException 问题2: overriding is disabled 问题3 :dev.langchain4j.exception.IllegalConfigurationException 1.4  langchain4j与springAI对比 1.1 简介 一个基于 Java 的库,旨在简化自然语言处理(NLP)和大型语言模型(LLM)

By Ne0inhk
【Java 开发日记】阻塞队列有哪些?拒绝策略有哪些?

【Java 开发日记】阻塞队列有哪些?拒绝策略有哪些?

目录 阻塞队列有哪些? 拒绝策略有哪些? 面试回答 阻塞队列有哪些? 在Java的java.util.concurrent包里面,阻塞队列的实现挺多的,我们可以根据它的功能和结构来记,主要分这么几类: 1. 按容量划分: * 有界队列: 就是队列有固定的容量。 * ArrayBlockingQueue: 最经典的一个,底层是数组,创建时必须指定大小。它的生产和消费用同一把锁,性能相对稳定。 * LinkedBlockingQueue: 底层是链表,它既可以是有界的(构造时指定容量),也可以默认是无界的(默认是Integer.MAX_VALUE,几乎相当于无界)。它的生产和消费用了两把锁,在高并发场景下吞吐量通常比ArrayBlockingQueue更高。 * 无界队列: 理论上是无限的,只要内存够就能一直放。 * PriorityBlockingQueue: 一个支持优先级排序的无界队列。元素必须实现Comparable接口,或者构造时传入Comparator。它出队的顺序是按优先级来的,不是先进先出 * DelayQueue: 一个很特殊的队

By Ne0inhk
Java网络聊天室——OverThinker-ChatRoom

Java网络聊天室——OverThinker-ChatRoom

—项目专栏— 🚀 Java Chatroom 实时聊天室系统 一个基于 Spring Boot 和 WebSocket 技术实现的轻量级实时聊天室项目。 ✨ 项目概述 这是一个采用 前后端分离 架构的 Web 聊天应用。它专注于提供一个稳定、实时的消息通信平台,支持用户认证、好友管理、以及核心的一对一私聊功能。 特性描述实时通信基于 WebSocket 实现,消息秒级推送。核心功能用户注册登录、好友列表、私聊会话、消息历史记录。后端架构Spring Boot 配合 MyBatis,快速构建 RESTful API。前端技术传统 HTML/CSS/JavaScript + jQuery,轻量易维护。 📸 界面展示 (Screenshots) 登录与注册 登录页面 注册页面 聊天主界面 ⚡ 项目体验说明 先看说明!

By Ne0inhk