【浅记】分而治之

归并排序

算法流程:

  1. 将数组A[1,n]排序问题分解为A[1,n/2]A[n/2+1,n]排序问题
  2. 递归解决子问题得到两个有序的子数组
  3. 将两个子数组合并为一个有序数组

符合分而治之的思想:

  1. 分解原问题
  2. 解决子问题
  3. 合并问题解

递归式求解

递归树法

用树的形式表示抽象递归

T ( n ) = { 2 T ( n 2 ) + O ( n ) , i f n > 1 O ( 1 ) , i f n = 1 T(n)=\begin{cases} 2T(\frac{n}{2})+O(n), &if&n>1\\ O(1),&if&n=1 \end{cases} T(n)={2T(2n)+O(n),O(1),ififn>1n=1

树的深度通常从0开始计,故层数等于n+1,后续统一用深度
可以得到,这个算法的时间复杂度是: T ( n ) = O ( n log ⁡ n ) T(n)=O(n\log n) T(n)=O(nlogn)

主定理法

对形如 T ( n ) = a T ( n b ) + f ( n ) T(n)=aT(\frac{n}{b})+f(n) T(n)=aT(bn)+f(n)的递归式:

  • 每个节点共a个分支
  • 每层以因子b速度下降
  • n log ⁡ b a n^{\log_ba} nlogba代表每层叶子节点代价之和

可以得到如下公式:
KaTeX parse error: {align} can be used only in display mode.
f ( n ) f(n) f(n)形式为 n k n^k nk时,可简化主定理公式:
KaTeX parse error: {align} can be used only in display mode.
例: T ( n ) = 5 T ( n 2 ) + n 3 T(n)=5T(\frac{n}{2})+n^3 T(n)=5T(2n)+n3

  1. k = 3 k=3 k=3
  2. a = 5 , b = 2 , log ⁡ b a = log ⁡ 2 5 a=5,b=2,\log_ba=\log_25 a=5,b=2,logba=log25
  3. k > log ⁡ b a k>\log_ba k>logba
  4. T ( n ) = Θ ( n 3 ) T(n)=\Theta(n^3) T(n)=Θ(n3)

最大子数组

基于枚举策略

蛮力枚举:枚举所有可能的情况
枚举 n + C n 2 n+C_n^2 n+Cn2种可能的区间 [ l , r ] , ( l < r ) [l,r],(l<r) [l,r],(l<r),求解最大子数组之和 S m a x S_{max} Smax,时间复杂度为 O ( n 3 ) O(n^3) O(n3)
优化枚举:蛮力枚举存在重复计算
S ( l , r ) = Σ i = 1 r X [ i ] = S ( l , r − 1 ) + X [ r ] S(l,r)=\Sigma_{i=1}^rX[i]=S(l,r-1)+X[r] S(l,r)=Σi=1rX[i]=S(l,r1)+X[r],时间复杂度为 O ( n 2 ) O(n^2) O(n2)

分而治之

  1. 将数组 X [ 1.. n ] X[1..n] X[1..n]分为 X [ 1.. n / 2 ] X[1..n/2] X[1..n/2] X [ n / 2 + 1.. n ] X[n/2+1..n] X[n/2+1..n]
  2. 递归求解子问题:
    1. S 1 S_1 S1:数组 X [ 1.. n / 2 ] X[1..n/2] X[1..n/2]的最大子数组
    2. S 2 S_2 S2:数组 X [ n / 2 + 1.. n ] X[n/2+1..n] X[n/2+1..n]的最大子数组
  3. 合并子问题,得到 S m a x S_{max} Smax
    1. S 3 S_3 S3:跨中点的最大子数组
    2. 数组 X X X的最大子数组之和 S m a x = m a x { S 1 , S 2 , S 3 } S_{max}=max\{S_1,S_2,S_3\} Smax=max{S1,S2,S3}

问题在于,如何高效地求解 S 3 S_3 S3

  1. m i d = n 2 mid=\frac n2 mid=2n
  2. S 3 S_3 S3可以分为左右两部分:
    1. L e f t Left Left:以 X [ m i d ] X[mid] X[mid]为结尾的最大子数组之和
    2. R i g h t Right Right:以 X [ m i d + 1 ] X[mid+1] X[mid+1]为开头的最大子数组之和
    3. S 3 = L e f t + R i g h t S_3=Left+Right S3=Left+Right

求解 S 3 S_3 S3的时间复杂度分析:

  1. 求解 L e f t Left Left:从 X [ m i d ] X[mid] X[mid]向前遍历求和,并记录最大值,时间复杂度为 O ( m i d ) O(mid) O(mid)
  2. 求解 R i g h t Right Right:从 X [ m i d + 1 ] X[mid+1] X[mid+1]向后遍历求和,并记录最大值,时间复杂度为 O ( n − m i d ) O(n-mid) O(nmid)
  3. 求解 S 3 S_3 S3的时间复杂度: O ( n ) O(n) O(n)

伪代码:

输入:数组X,数组下标low,high
输出:最大子数组之和Smax
if low=high then
|	return X[low]
end
else
|	mid	<-	(low+high)/2
|	S1	<-	MaxSubArray(X,low,mid)
|	S2	<-	MaxSubArray(X,mid+1,high)
|	S3	<-	CrossingSubArray(X,low,mid,high)
|	Smax	<-	Max(S1,S2,S3)
|	return Smax
end

时间复杂度分析

$T(n)=
\left{
\begin{align}
&1,\quad&n=1\
&2T(\frac n2)+O(n),\quad&n>1
\end{align}
\right.
\$
按照上面提到的递归树求解的方法,可以得到: T ( n ) = n log ⁡ n T(n)=n\log n T(n)=nlogn

逆序对计数

逆序对:当 i < j i<j i<j A [ i ] > A [ j ] A[i]>A[j] A[i]>A[j]的二元组 ( A [ i ] , A [ j ] ) (A[i],A[j]) (A[i],A[j])

蛮力枚举

对于数组的每个元素 A [ i ] A[i] A[i],枚举 j ( j > i ) j(j>i) j(j>i),并统计逆序对数目。
伪代码:

输入:一个大小为n的数组A[1..n]
输出:数组A[1..n]中逆序对数目Sum
Sum	<-	0
for i <- 1 to n do
|	for j <- i+1 to n do
|	 |	if A[i]>A[j] then
|	 |	|	Sum <- Sum+1
|	 |	end
|	end
end
return Sum

这一算法时间复杂度为 O ( n 2 ) O(n^2) O(n2)

分而治之

  1. 将数组 A [ 1.. n ] A[1..n] A[1..n]分为 A [ 1.. n 2 ] A[1..\frac n2] A[1..2n] A [ n 2 + 1.. n ] A[\frac n2+1..n] A[2n+1..n]
  2. 递归求解子问题
    1. S 1 S_1 S1:仅在 A [ 1.. n 2 ] A[1..\frac n2] A[1..2n]中的逆序对数目
    2. S 2 S_2 S2:仅在 A [ n 2 + 1.. n ] A[\frac n2+1..n] A[2n+1..n]中的逆序对数目
  3. 合并 A [ 1.. n ] A[1..n] A[1..n]分为 A [ 1.. n 2 ] A[1..\frac n2] A[1..2n] A [ n 2 + 1.. n ] A[\frac n2+1..n] A[2n+1..n]的解
    1. S 3 S_3 S3:跨越子数组的逆序对数目
    2. S = S 1 + S 2 + S 3 S=S_1+S_2+S_3 S=S1+S2+S3

策略一:直接求解

对每个 A [ j ] ∈ A [ m + 1 , n ] A[j]\in A[m+1,n] A[j]A[m+1,n],枚举 A [ i ] ∈ A [ 1.. m ] A[i]\in A[1..m] A[i]A[1..m]并统计逆序对数目。
求解 S 3 S_3 S3的算法运行时间: O ( n 2 ) O(n^2) O(n2)
分而治之框架的运行时间: T ( n ) = 2 T ( n 2 ) + O ( n 2 ) T(n)=2T(\frac n2)+O(n^2) T(n)=2T(2n)+O(n2)
直接求解的分而治之较蛮力枚举并未提高算法运行时间。

  • 运行时间受制于跨越子数组的逆序对计数方法
  • 数组的有序性通常有助于提高算法的运行时间

策略二:排序求解

  1. 分别对数组 A [ 1.. m ] A[1..m] A[1..m] A [ m + 1.. n ] A[m+1..n] A[m+1..n]进行排序
  2. 对于每个 A [ j ] ∈ A [ m + 1.. n ] A[j]\in A[m+1..n] A[j]A[m+1..n],采用二分查找为其在 A [ 1.. m ] A[1..m] A[1..m]中定位
  3. A [ j ] A[j] A[j] A [ 1.. m ] A[1..m] A[1..m]定位点右侧的元素均可与 A [ j ] A[j] A[j]构成逆序对

求解 S 3 S_3 S3的算法运行时间: O ( n log ⁡ 2 n ) O(n\log^2 n) O(nlog2n)
分治框架的算法运行时间: T ( n ) = 2 T ( n 2 ) + O ( n log ⁡ n ) T(n)=2T(\frac n2)+O(n\log n) T(n)=2T(2n)+O(nlogn)

算法优化

合并问题解的同时对数组进行排序

  • 归并过程中可同时计算逆序对数目

策略三:归并求解

  1. 从左到右扫描有序子数组: A [ i ] ∈ A [ 1.. m ] , A [ j ] ∈ A [ m + 1.. n ] A[i]\in A[1..m],A[j]\in A[m+1..n] A[i]A[1..m],A[j]A[m+1..n]
    1. 如果 A [ i ] > A [ j ] A[i]>A[j] A[i]>A[j],统计逆序对, j j j向右移
    2. 如果 A [ i ] < A [ j ] A[i]<A[j] A[i]<A[j] i i i向右移
  2. 利用归并排序框架保证合并后数组的有序性

伪代码:

输入:数组A[1..n],数组下标left,right
输出:数组A[left..right]的逆序对数,递增数组A[left..right]
if left >= right then
|	return A[left..right]
end
mid	<-(left+right)/2
S1	<- CountInver(A,left,mid)
S2	<- CountInver(A,mid+1,right)
S3	<- MergeCount(A,left,mid,right)
S		<- S1+S2+S3
return S,A[left..right]

时间复杂度: T ( n ) = O ( n log ⁡ n ) T(n)=O(n\log n) T(n)=O(nlogn)

快速排序

归并排序:简化分解,侧重合并
快速排序:侧重分解,简化合并

数组划分

  1. 任选 x x x作为分界线,称为主元
  2. 交换重排,满足 x x x左侧元素小于右侧

实现方法:

  1. 选取固定位置主元 x x x,如尾元素
  2. 维护两个指针变量,分别指向两个部分的右端点
  3. 考察数组元素 A [ j ] A[j] A[j],只和主元比较
    1. A [ j ] ≤ x A[j]\leq x A[j]x,则交换 A [ j ] A[j] A[j] A [ i ] + 1 A[i]+1 A[i]+1 i i i j j j右移
    2. A [ j ] > x A[j]> x A[j]>x,则 j j j右移
  4. 把主元放在中间作分界线

伪代码Partition(A,p,r)

输入:数组A,起始位置p,终止位置r
输出:划分位置q
x <- A[r]//选取主元
i <- p-1
for j <- p to r-1 do| if A[j] <= x then| | exchange A[i+1] with A[j]| | i <- i+1| end
end
exchange A[i+1] with A[r]
q <- i+1
return q

快速排序的伪代码

输入:数组A,起始位置p,终止位置r
输出:有序数组A
if p<r then
| q <- Partition(A,p,r)
| QuickSort(A,p,q-1)
| QuickSort(A,q+1,r)
end

最好情况: O ( n log ⁡ n ) O(n\log n) O(nlogn)
最坏情况: O ( n 2 ) O(n^2) O(n2)
快速排序看似不优于归并排序。

次序选择问题

输入:

  • 包含 n n n个不同元素的数组 A [ 1.. n ] A[1..n] A[1..n]
  • 整数 k k k

输出:

  • 数组 A [ 1.. n ] A[1..n] A[1..n]中第 k k k小的元素 ( 1 ≤ k ≤ n ) (1\leq k\leq n) (1kn)

固定位置划分求解

image.png
选取固定位置主元,小于主元的元素个数 q − p q-p qp

  • 情况1: k = q − p + 1 k=q-p+1 k=qp+1 A [ q ] A[q] A[q]为数组第 k k k小元素
  • 情况2: k < q − p + 1 k<q-p+1 k<qp+1,在 A [ p . . q − 1 ] A[p..q-1] A[p..q1]中寻找第 k k k小元素
  • 情况3: k > q − p + 1 k>q-p+1 k>qp+1,在 A [ q + 1.. r ] A[q+1..r] A[q+1..r]中寻找第 k − ( q − p + 1 ) k-(q-p+1) k(qp+1)小元素

子问题始终唯一,无需合并问题解。

伪代码

输入:数组A,起始位置p,终止位置r,元素次序k
输出:第k小元素x
q <- Partition(A,p,r)
if k=(q-p+1) then
| x <- A[q]
end
if k<(q-p+1) then
| x <-Selection(A,p,q-1,k)
end
if k>(q-p+1) then
| x <- Selection(A,q+1,r,k-(q-p+1))
end
return x

复杂度分析:
最好情况: T ( n ) = Ω ( n ) T(n)=\Omega(n) T(n)=Ω(n)
期望时间复杂度: T ( n ) = Θ ( n ) T(n)=\Theta(n) T(n)=Θ(n)
最坏情况: T ( n ) = O ( n 2 ) T(n)=O(n^2) T(n)=O(n2)

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

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

7.网络原理之TCP_IP(下)

文章目录 4.传输层重点协议4.1TCP协议4.1.1TCP协议段格式4.1.2TCP原理4.1.2.1确认应答机制 ACK&#xff08;安全机制&#xff09;4.1.2.2超时重传机制&#xff08;安全机制&#xff09;4.1.2.3连接管理机制&#xff08;安全机制&#xff09;4.1.2.4滑动窗口&#xff08;效率机制…

uni-app:实现元素在屏幕中的居中(绝对定位absolute)

一、实现水平居中 效果 代码 <template><view><view class"center">我需要居中</view></view> </template><style>.center {position: absolute;left:50%;transform: translateX(-50%);border:1px solid black;} </s…

freertos简介与移植

freertos是一个可裁剪的小型rtos系统&#xff0c;特点&#xff1a; 支持抢占式&#xff0c;合作式和时间片调度saferos衍生自freertos&#xff0c;更完整提供了一个用于低功耗的tickless模式系统的组件在创建时可以选择动态或者静态的ram&#xff0c;例如任务&#xff0c;消息…

快排三种递归及其优化,非递归和三路划分

个人主页&#xff1a;Lei宝啊 愿所有美好如期而遇 目录 快排简介&#xff1a; 快排的三种递归实现&#xff1a; Hoare&#xff1a; 挖坑&#xff1a; 双指针&#xff1a; 小区间优化&#xff1a; 三数取中优化&#xff1a; 快排非递归实现&#xff1a; 快排的三路划…

计算机网络(一):概述

参考引用 计算机网络微课堂-湖科大教书匠计算机网络&#xff08;第7版&#xff09;-谢希仁 1. 计算机网络在信息时代的作用 计算机网络已由一种通信基础设施发展成为一种重要的信息服务基础设施计算机网络已经像水、电、煤气这些基础设施一样&#xff0c;成为我们生活中不可或…

《CTFshow-Web入门》10. Web 91~110

Web 入门 索引web91题解总结 web92题解总结 web93题解 web94题解 web95题解 web96题解 web97题解 web98题解 web99题解总结 web100题解 web101题解 web102题解 web103题解 web104题解 web105题解总结 web106题解 web107题解 web108题解 web109题解 web110题解 ctf - web入门 索…

红外遥控器 数据格式,按下及松开判断

红外遥控是一种无线、非接触控制技术&#xff0c;具有抗干扰能力强&#xff0c;信息传输可靠&#xff0c;功耗低&#xff0c;成本低&#xff0c;易实现等显著优点&#xff0c;被诸多电子设备特别是家用电器广泛采用&#xff0c;并越来越多的应用到计算机系统中。 同类产品的红…

iOS 视频压缩 mov转mp4 码率

最近还是因为IM模块的功能&#xff0c;IOS录制MOV视频发送后&#xff0c;安卓端无法播放&#xff0c;迫不得已兼容将MOV视频转为MP4发送。 其中mov视频包括4K/24FPS、4K/30FPS、4K/60FPS、720p HD/30FPS、1080p HD/30FPS、1080p HD/60FPS&#xff01; 使用AVAssetExportSessi…

使用sqlmap获取数据步骤

文章目录 1.使用sqlmap获取所有数据库2.使用sqlmap获取当前连接数据库3.使用sqlmap获取当前数据库下所有表名4.使用sqlmap获取当前数据库下某个表下所有列名5.使用sqlmap获取当前数据库下某个表下指定字段的数据6.测试当前用户是否是管理员7.使用burpsqlmap批量检测8.脱库命令9…

zkLogin构建者的最佳实践和业务思考

随着zkLogin在Sui主网上线&#xff0c;构建者可以开始为其应用程序提供丝滑的帐户创建服务。与任何新技术集成一样&#xff0c;构建者需要考虑许多重要的问题&#xff0c;以降低风险并成功优化。 本文概述了其中一些考虑因素&#xff0c;并突出了zkLogin文档中提供的明确指导。…

二、机器学习基础知识:Python数据处理基础

文章目录 1、基本数据类型1.1 数字类型&#xff08;Number&#xff09;1.2 字符串类型&#xff08;String&#xff09;1.3 列表类型&#xff08;List&#xff09;1.4 元组类型&#xff08;Tuple&#xff09;1.5 字典类型&#xff08;Dictionary&#xff09;1.6 集合类型&#x…

掌动智能:替代JMeter的压力测试工具有哪些

JMeter是一个广泛使用的开源压力测试工具&#xff0c;但在实际应用中&#xff0c;也有一些其他优秀的替代品可供选择。本文将介绍几个可替代JMeter的压力测试工具&#xff0c;它们在功能、性能和易用性方面都具有独特优势&#xff0c;可以满足不同压力测试需求的选择。 一、Gat…

lvgl不能显示图片,但可以显示按键?

AT32F403A, IAR, ST7735S, LVGL8.3。 一、现象&#xff1a; 本来想着用LVGL做一个摄像头的显示功能 切换动态。可是死活实现不了功能。 因为移植好LVGL后&#xff0c;首先测试了显示按键&#xff0c;功能正常&#xff0c;以为是一切正常。在模拟器上调试效果完成后&#xf…

Scala第十三章节

Scala第十三章节 1. 高阶函数介绍 2. 作为值的函数 3. 匿名函数 4. 柯里化 5. 闭包 6. 控制抽象 7. 案例: 计算器 scala总目录 文档资料下载

IDEA git操作技巧大全,持续更新中

作者简介 目录 1.创建新项目 2.推拉代码 3.状态标识 5.cherry pick 6.revert 7.squash 8.版本回退 9.合并冲突 1.创建新项目 首先我们在GitHub上创建一个新的项目&#xff0c;然后将这个空项目拉到本地&#xff0c;在本地搭建起一个maven项目的骨架再推上去&#xff0…

Python集成开发环境(IDE):WingPro for Mac

WingPro for Mac是一款Python集成开发环境&#xff08;IDE&#xff09;软件&#xff0c;它提供了一系列强大的工具和功能&#xff0c;帮助Python开发人员提高开发效率和质量。 WingPro for Mac拥有直观的用户界面和强大的调试器&#xff0c;可以帮助用户快速定位问题和修复错误…

12链表-双指针

目录 LeetCode之路——21. 合并两个有序链表 分析&#xff1a; LeetCode之路——19. 删除链表的倒数第 N 个结点 分析&#xff1a; LeetCode之路——21. 合并两个有序链表 将两个升序链表合并为一个新的 升序 链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的…

宝塔反代openai官方API接口详细教程,502 Bad Gateway问题解决

一、前言 宝塔反代openai官方API接口详细教程&#xff0c;实现国内使用ChatGPT502 Bad Gateway问题解决&#xff0c; 此方法最简单快捷&#xff0c;没有复杂步骤&#xff0c;不容易出错&#xff0c;即最简单&#xff0c;零代码、零部署的方法。 二、实现前提 一台海外VPS服务…

网络安全——自学(黑客)方法

如果你想自学网络安全&#xff0c;首先你必须了解什么是网络安全&#xff01;&#xff0c;什么是黑客&#xff01;&#xff01; 1.无论网络、Web、移动、桌面、云等哪个领域&#xff0c;都有攻与防两面性&#xff0c;例如 Web 安全技术&#xff0c;既有 Web 渗透2.也有 Web 防…

云原生之使用Docker部署RSS阅读器Huntly

云原生之使用Docker部署RSS阅读器Huntly 一、Huntly介绍1.1 Huntly简介1.2 Huntly功能 二、本次实践规划2.1 本地环境规划2.2 本次实践介绍 三、本地环境检查3.1 检查Docker服务状态3.2 检查Docker版本3.3 检查docker compose 版本 四、下载Huntly镜像五、部署Huntly5.1 创建挂…