分治法之选择问题 (1)
NOIP基础算法讲解2
while(i<=mid)do begin temp[p]:=a[i];inc(p);inc(i);end while(j<=right)do begin temp[p]:=a[j];inc(p);inc(i);end for i:=left to right do a[i]:=temp[i]; end;
数据范围:n≤10^6。所有数之和不超过10^9。
例题8:快速幂
【问题】计算an mod k的值 ,其中n<=109。
方法1:朴素算法。每次乘以a,时间复杂度O(n)。 function power(a,n:longint):longint; var x:longint; begin x:=1; for i:=1 to n do x:=x*a; power:=x; end;
时间效率不尽如人意….. 问题出现在哪里呢??
方法2:分治策略
采用分治求解: ➢划分问题:把序列分成元素个数尽量相等的两半; ➢递归求解:统计i和j均在左边或者均在右边的逆序对个数; ➢合并问题:统计i在左边,但j在右边的逆序对个数。
记数列a[st]~a[ed]的逆序对数目为d(st,ed); mid=(st+ed)/2,则有:
三、分治的三步骤
①划分问题:将要解决的问题分解成若干个规模较 小的同类子问题;
②递归求解:当子问题划分得足够小时,求解出子 问题的解。
③合并问题:将子问题的解逐层合并成原问题的解。
四、分治的框架结构
procedure Divide() begin
分治法解决问题的步骤
分治法解决问题的步骤一、基础概念类题目(1 - 5题)题目1:简述分治法解决问题的基本步骤。
解析:分治法解决问题主要有三个步骤:1. 分解(Divide):将原问题分解为若干个规模较小、相互独立且与原问题形式相同的子问题。
例如,对于排序问题,可将一个大的数组分成两个较小的子数组。
2. 求解(Conquer):递归地求解这些子问题。
如果子问题规模足够小,则直接求解(通常是一些简单的基础情况)。
对于小到只有一个元素的子数组,它本身就是有序的。
3. 合并(Combine):将各个子问题的解合并为原问题的解。
在排序中,将两个已排序的子数组合并成一个大的有序数组。
题目2:在分治法中,分解原问题时需要遵循哪些原则?解析:1. 子问题规模更小:分解后的子问题规模要比原问题小,这样才能逐步简化问题。
例如在归并排序中,不断将数组对半分,子数组的长度不断减小。
2. 子问题相互独立:子问题之间应该尽量没有相互依赖关系。
以矩阵乘法的分治算法为例,划分后的子矩阵乘法之间相互独立进行计算。
3. 子问题与原问题形式相同:方便递归求解。
如二分查找中,每次查找的子区间仍然是一个有序区间,和原始的有序区间查找问题形式相同。
题目3:分治法中的“求解”步骤,如果子问题规模小到什么程度可以直接求解?解析:当子问题规模小到可以用简单的、直接的方法(如常量时间或线性时间复杂度的方法)解决时,就可以直接求解。
例如,在求数组中的最大最小值问题中,当子数组只有一个元素时,这个元素既是最大值也是最小值,可以直接得出结果。
题目4:分治法的“合并”步骤有什么重要性?解析:1. 构建完整解:它将各个子问题的解组合起来形成原问题的解。
例如在归并排序中,单独的两个子数组排序好后,只有通过合并操作才能得到整个数组的有序排列。
2. 保证算法正确性:如果合并步骤不正确,即使子问题求解正确,也无法得到原问题的正确答案。
例如在分治算法计算斐波那契数列时,合并不同子问题的结果来得到正确的斐波那契数是很关键的。
分治法
顾铁成
1
引例:称硬币
如果给你一个装有16枚硬币的袋子,其中有一
枚是假的,并且其重与真硬币不同。你能不能 用最少的比较次数,找出这个假币?
为了帮助你完成这个任务,将提供一台可用来 比较两组硬币重量的仪器,利用这台仪器,可
以知道两组硬币的重量是否相同。
2
引例:称硬币
常规的解决方法是先将这些硬币分成两
15
当 k = 1 时,各种可能的残缺棋盘
16
三格板的四个不同方向
17
【输入】
第一行输入棋盘 的总行数,第二 行输入残缺棋盘 的格子坐标。
【样例输入】 4
4 1
【样例输出】 2 2 3 3 2 1 1 3 4 4 1 5
【输出】
覆盖的矩阵图。
0 4 5 5
18
问题分析
很明显,当K=0时,是不需要三格板的,而当
24
【样例输入】 5 3 23 8 91 56 4 【样例输出】 1
25
问题分析
对于一组混乱无序的数来说,要找到第k
小的元素,通常要经过两个步骤才能实 现:
第一步:将所有的数进行排序; 第二步:确定第k个位置上的数。
26
问题分析
传统的排序算法(插入排序、选择排序
、冒泡排序等)大家都已经很熟悉了, 但已学过的排序方法无论从速度上பைடு நூலகம்还 是从稳定性方面,都不是最佳的。
将7作为一个参照数;
将这个数组中比7大的数放在7的左边; 比7大的数放在7的右边;
这样,我们就可以得到第一次数组的调整:
[ 4 2 6 6 1 ] 7 [ 10 22 9 8 ]
29
用分治法解决问题
分治法所能解决的问题具有以下几个特征: 分治法所能解决的问题具有以下几个特征:
1.该问题的规模缩小到一定的程度就可以容易地解决; 1.该问题的规模缩小到一定的程度就可以容易地解决; 该问题的规模缩小到一定的程度就可以容易地解决 2.该问题可以分解为若干个规模较小且基本相同的子问 2.该问题可以分解为若干个规模较小且基本相同的子问 题。 3.利用该问题分解出的子问题的解可以合并为该问题的 3.利用该问题分解出的子问题的解可以合并为该问题的 解;
分治策略解决问题
问题1 问题1:找出伪币
一个装有1 6枚硬币的袋子 一个装有1 6枚硬币的袋子,1 6枚硬币中有一个 枚硬币的袋子, 6枚硬币中有一个 是伪造的, 是伪造的,并且那个伪造的硬币比真的硬币要轻 一些。你的任务是找出这枚伪造的硬币。 一些。你的任务是找出这枚伪造的硬币。 为了帮助你完成这一任务, 为了帮助你完成这一任务,将提供一台可用来比 较两组硬币重量的仪器,比如天平。 较两组硬币重量的仪器,比如天平。利用这台仪 可以知道两组硬币的重量是否相同。 器,可以知道两组硬币的重量是否相同。
找金块的示例图
方法2 方法2:
n≤2,识别出最重和最轻的金块,一次比较就足够 ≤2,识别出最重和最轻的金块, 了。 n>2,第一步,把这袋金块平分成两个小袋A和B。 第一步,把这袋金块平分成两个小袋A 第二步,分别找出在A 中最重和最轻的金块。 第二步,分别找出在A和B中最重和最轻的金块。设 A中最重和最轻的金块分别为HA 与LA,以此类推, 中最重和最轻的金块分别为HA LA,以此类推, B中最重和最轻的金块分别为HB 和LB。第三步, 中最重和最轻的金块分别为HB LB。第三步, 通过比较HA HB,可以找到所有金块中最重的; 通过比较HA 和HB,可以找到所有金块中最重的; 通过比较LA LB,可以找到所有金块中最轻的。 通过比较LA 和LB,可以找到所有金块中最轻的。 在第二步中, 则递归地应用分而治之方法。 在第二步中,若n>2,则递归地应用分而治之方法。
算法设计与分析课件--分治法-线性时间选择
2.5 线性时间选择
这样找到的m*划分是否能达到O(n)的时间复杂度? |A| = |D| = 2r, |B| = |C| = 3r +2,n = 10r +5. |A| + |D| + |C| = 7r + 2 = 7(n-5)/10 +2 = 7n/10 -1.5 < 7n/10 表明子问题的规模不超过原问题的7/10(d)。
T(n) = T(cn) + T(dn) + tn
6
2.5 线性时间选择
Select(S, k) Input: n个数的数组S,正整数k
T(n) = T(cn) + T(dn) + tn
Output: S中的第k个小元素
1. 将S划分成5个元素一组,共[n/5]个组;
2. 每组寻找一个中位数,把这些中位数放到集合M中;
寻找一个分割点m*, 使得左边子表S1中的元素都小于m*, 右子表 S2中的元素都大于m*。 如果寻找m*的时间复杂度达到O(nlogn), 那就不如直接使用排序 算法了。 如果直接寻找m*, 时间复杂度是O(n). 假设选择算法的时间复杂度为T(n), 递归调用这个算法在S的一 个真子集M上寻找m*,应该使用T(cn)时间,这里c是小于1的常数, 反映了M的规模与S相比缩小许多。
✓ 不妨假设n是5的倍数,且n/5是奇数,即n/5 = 2r+1. 于是: |A| = |D| = 2r, |B| = |C| = 3r +2,n = 10r +5.
✓ 如果A和D中的元素都小于m*,那么把它们的元素都加入到S1, S1对应规约后子问题的上限。 类似的,若A和D中的元素都 大于m*, 则把他们的元素都加 入到S2,S2对应规约后子问题 的上限。
算法设计与分析-分治法
3.2.1 归并排序
算法3.1——归并排序
void MergeSort(int r[ ], int r1[ ], int s, int t) {
if (s= =t) r1[s]=r[s]; //只有一个元素,直接赋值 else {
m=(s+t)/2; Mergesort(r, r1, s, m); //归并排序前半个子序列 Mergesort(r, r1, m+1, t); //归并排序后半个子序列 Merge(r1, r, s, m, t); //合并两个已排序的子序列 } }
A、B、C、D 四个区域
Ø想法
Ø 用二维数组data[N][N]表示N×N的方阵,观察方阵中数
字的规律,可以从外层向里层填数。 Ø 设变量size表示方阵的大小,则初始时size = N,填完一
层则size = size - 2;
Ø想法
Ø 设变量begin表示每一层的起始位置,变量i和j分别表示
MergeSort(r,r1,1,1) r1[1]=r[1]
Merge(r1,r,0,0,1)
MergeSort(r,r1,2,3)
MergeSort(r,r1,2,2) r1[2]=r[2]
MergeSort(r,r1,3,3) r1[3]=r[3]
Merge(r1,r,2,2,3)
Merge(r1,r,0,1,3)
• 分治思想 • 归并排序 • 快速排序 • 折半查找 • 选择问题 • 最大子段和问题 • 棋盘覆盖问题 • 循环赛日程安排问题
3.1 基本思想 3.2 排序问题中的分治算法 3.3 查找问题中的分治算法 3.4 组合问题中的分治算法 3.5 典型问题的C++程序(略)
实验1 分治法找到数组元素中的最大值与最小值
算法分析与设计实验报告第 1 次实验附录:完整代码#include <time.h>#include <iostream>#include <iomanip>#include <stdlib.h>using namespace std;void min_max(int a[],int i,int j,int &min,int &max) {int mid,max1,max2,min1,min2;if(i==j){max=a[i];min=a[i];return;}if(j==i+1){if(a[i]>a[j]){min=a[j];max=a[i];}else{min=a[i];max=a[j];}}else{mid=(i+j)/2;min_max(a,i,mid,min1,max1);min_max(a,mid+1,j,min2,max2);if(min1>min2)min=min2;elsemin=min1;if(max1>max2)max=max1;elsemax=max2;}}int main (){int m,a[100],min,max;while(1){int f;cout<<"随机数组的规模:";cin>>m;cout<<"随机数的范围:";cin>>f;//计时开始clock_t start,end,over;start=clock();end=clock();over=end-start;start=clock();srand((unsigned)time(NULL));for(int i=1;i<=m;i++){a[i]=(rand()%(f)+0);cout<<a[i]<<' ';}cout<<endl;min_max(a,1,m,min,max);cout<<"最小值:"<<min<<endl;cout<<"最大值:"<<max<<endl;end=clock();printf("The time is %6.3f",(double)(end-start-over)/CLK_TCK);cout<<endl;cout<<endl;}}。
分治法练习题
分治法练习题分治法是一种常见的算法设计方法,其核心思想是将问题划分成若干个规模较小且结构相似的子问题,然后分别解决这些子问题,最后将子问题的结果合并得到原问题的解。
在实际应用中,选取合适的问题划分方式以及合并子问题的结果是非常关键的。
下面,我将为您介绍两个分治法的练习题。
题目一:寻找最大子数组和给定一个整数数组,找到其连续子数组中的最大和。
例如,输入数组[-2, 1, -3, 4, -1, 2, 1, -5, 4],其最大子数组和为6,对应的子数组为[4, -1, 2, 1]。
解题思路:1. 将原问题划分成规模较小的子问题:将数组分为两部分,分别求解左子数组和右子数组的最大子数组和,以及跨越中点的最大子数组和。
2. 递归求解子问题:对于左右子数组,可以再次使用分治法求解;对于跨越中点的最大子数组和,可以通过以中点为中心,向左右扩展来得到。
3. 合并子问题的结果:对于左右子数组的最大子数组和,取较大值作为整个数组的最大子数组和;对于跨越中点的最大子数组和,取两边相加的最大值。
题目二:求解逆序对个数给定一个数组,逆序对是指数组中两个元素a[i]和a[j],满足i < j且a[i] > a[j]。
请设计一个算法,求解给定数组中逆序对的个数。
解题思路:1. 将原问题划分成规模较小的子问题:将数组平均分为两部分,分别求解左子数组和右子数组中逆序对的个数,以及两个子数组之间的逆序对个数。
2. 递归求解子问题:对于左右子数组,可以再次使用分治法求解;对于两个子数组之间的逆序对个数,可以通过归并排序的思想来求解。
3. 合并子问题的结果:将左右子数组合并为一个有序数组,并统计两个子数组之间的逆序对个数。
同时,递归返回的结果也需要累加进逆序对的总数。
通过以上两个练习题,我们可以更加深入地理解和应用分治法这一算法设计思想,同时也能提升对问题分解和结果合并的能力。
当然,在实际应用中,我们需要灵活运用分治法以及结合具体问题来设计合适的算法,并注意算法的效率和性能。
算法分析作业
算法分析作业 IMB standardization office【IMB 5AB- IMBK 08- IMB 2C】算法分析练习题(一)一、选择题1、二分搜索算法是利用(A )实现的算法。
A、分治策略B、动态规划法C、贪心法D、回溯法2、下列不是动态规划算法基本步骤的是(A )。
A、找出最优解的性质B、构造最优解C、算出最优解D、定义最优解3.下列算法中通常以自底向上的方式求解最优解的是(B )。
A、备忘录法B、动态规划法C、贪心法D、回溯法4、衡量一个算法好坏的标准是(C )。
A 运行速度快B 占用空间少C 时间复杂度低D 代码短5、以下不可以使用分治法求解的是(D )。
A 棋盘覆盖问题B 选择问题C 归并排序D 0/1背包问题6. 实现循环赛日程表利用的算法是(A )。
A、分治策略B、动态规划法C、贪心法D、回溯法7.备忘录方法是那种算法的变形。
( B )A、分治法B、动态规划法C、贪心法D、回溯法8.最长公共子序列算法利用的算法是(B )。
A、分支界限法B、动态规划法C、贪心法D、回溯法9.实现棋盘覆盖算法利用的算法是(A )。
A、分治法B、动态规划法C、贪心法D、回溯法10. 矩阵连乘问题的算法可由(B)设计实现。
A、分支界限算法B、动态规划算法C、贪心算法D、回溯算法11、Strassen矩阵乘法是利用(A )实现的算法。
A、分治策略B、动态规划法C、贪心法D、回溯法12、使用分治法求解不需要满足的条件是(A )。
A 子问题必须是一样的B 子问题不能够重复C 子问题的解可以合并D 原问题和子问题使用相同的方法解13、下列算法中不能解决0/1背包问题的是(A )A 贪心法B 动态规划C 回溯法D 分支限界法14.实现合并排序利用的算法是(A )。
A、分治策略B、动态规划法C、贪心法D、回溯法15.下列是动态规划算法基本要素的是(D )。
A、定义最优解B、构造最优解C、算出最优解D、子问题重叠性质16.下列算法中通常以自底向下的方式求解最优解的是(B )。
算法练习题-分章节-带答案
算法练习题-分章节-带答案一、选择题1、下面关于算法的描述,正确的是()A、一个算法只能有一个输入B、算法只能用框图来表示C、一个算法的执行步骤可以是无限的D、一个完整的算法,不管用什么方法来表示,都至少有一个输出结果2、一位爱好程序设计的同学,想通过程序设计解决“韩信点兵”的问题,他制定的如下工作过程中,更恰当的是()A、设计算法,编写程序,提出问题,运行程序,得到答案B、分析问题,编写程序,设计算法,运行程序,得到答案C、分析问题,设计算法,编写程序,运行程序,得到答案D、设计算法,提出问题,编写程序,运行程序,得到答案3、下面说法正确的是()A、算法+数据结构=程序B、算法就是程序C、数据结构就是程序D、算法包括数据结构4、衡量一个算法好坏的标准是()。
A、运行速度快B、占用空间少C、时间复杂度低D、代码短5、解决一个问题通常有多种方法。
若说一个算法“有效”是指()。
A、这个算法能在一定的时间和空间资源限制内将问题解决B、这个算法能在人的反应时间内将问题解决C、这个算法比其他已知算法都更快地将问题解决D、A和C6、算法分析中,记号O表示(),记号表示()。
A.渐进下界B.渐进上界C.非紧上界D.非紧下界7、以下关于渐进记号的性质是正确的有:()A.f(n)(g(n)),g(n)(h(n))f(n)(h(n))B.f(n)O(g(n)),g(n)O(h(n))h(n)O(f(n))C.O(f(n))+O(g(n))=O(min{f(n),g(n)})D.f(n)O(g(n))g(n)O(f(n))8、记号O的定义正确的是()。
A.O(g(n))={f(n)|存在正常数c和n0使得对所有nn0有:0f(n)cg(n)};B.O(g(n))={f(n)|存在正常数c和n0使得对所有nn0有:0cg(n)f(n)};C.O(g(n))={f(n)|对于任何正常数c>0,存在正数和n0>0使得对所有nn0有0f(n)D.O(g(n))={f(n)|对于任何正常数c>0,存在正数和n0>0使得对所有nn0有:0cg(n)<f(n)};9、记号的定义正确的是()。
分冶法
f ( n) n 1 ( n ), d 1
d
T (n) 2T (n / 2) ( n 1)
a b 2, d 1, a b
d
d
T ( n) ( n lg n)
解递推方程得精确解: T (n) n log2 n n 1
10:39
13/42
减一技术
原问题(规模 n) (1) 子问题(n-1) (n-1)解 (n) 解 扩展解
无解
折 半 查 找
原问题解
插 入 排 序
10:39
22/42
插入排序(Insertion Sort)
任务:对 n 个元素作插入排序(规模 n) 减一策略 ——自顶向下:规模减小 ① 规模减小:规模减一,即 n-1 ② 求解:解 n-1规模子问题 ③ 扩展解:n-1规模解扩展为 n规模解 扩展方法的不同,有不同的插入排序 减一过程递归进行,直到 规模 = 1或0 为止 实现方法 —— 自底向上:规模增大 为便于实现,规模从 0 或 1 增加到 n
1, n 2 T ( n) k 2T ( n / 2) 1, n 2
10:39
7/42
分治法的一般时间效率分析 规模 n , 每次分为 a 个子问题,子问题规模相等 n/b 为简化分析, 不妨设 n = bk, k = 1, 2, 3, ...
通用分治递推式
c , n t 常量时间(基本操作次数) T ( n) aT ( n / b) f ( n), n t , a 1, b 2, c 0 f (n) : 分解时间 + 合并时间
平均 Tavg (n) 2n ln n 1.38n log2 n (n log2 n) 效率
分治法例题
选择题
下列哪个问题适合使用分治法解决?
A. 求解一元二次方程
B. 求解线性方程组
C. 归并排序(正确答案)
D. 求解最短路径问题(如Dijkstra算法)
分治法在解决问题时,通常会将问题划分为:
A. 一个更小的问题和一个辅助问题
B. 两个或更多个相似的子问题(正确答案)
C. 多个完全不相关的问题
D. 一个更大的问题和一个简化的问题
使用分治法求解时,递归的终止条件通常是:
A. 问题规模达到预设的阈值(正确答案)
B. 问题变得无法再分解
C. 找到问题的精确解
D. 递归深度达到某一值
在分治法中,将问题划分为子问题后,通常需要对子问题的解进行:
A. 丢弃
B. 合并(正确答案)
C. 忽略
D. 重新排序
下列哪个问题可以通过分治法递归地求解?
A. 计算数组的平均值
B. 查找数组中的最大值
C. 计算斐波那契数列的第n项(正确答案)
D. 判断一个数是否为质数
分治法的时间复杂度通常可以通过什么来分析?
A. 递归树(正确答案)
B. 动态规划表
C. 贪心策略
D. 暴力枚举
下列哪个步骤不是分治法的一般过程?
A. 分解
B. 解决
C. 跳过(正确答案)
D. 合并
使用分治法解决最大子序和问题时,通常会将数组划分为:
A. 左右两个子数组(正确答案)
B. 前半部分和后半部分
C. 奇数索引和偶数索引部分
D. 任意两个不相交的子数组。
分治算法详解及经典例题
分治算法详解及经典例题⼀、基本概念在计算机科学中,分治法是⼀种很重要的算法。
字⾯上的解释是“分⽽治之”,就是把⼀个复杂的问题分成两个或更多的相同或相似的⼦问题,再把⼦问题分成更⼩的⼦问题……直到最后⼦问题可以简单的直接求解,原问题的解即⼦问题的解的合并。
这个技巧是很多⾼效算法的基础,如排序算法(快速排序,归并排序),傅⽴叶变换(快速傅⽴叶变换)……任何⼀个可以⽤计算机求解的问题所需的计算时间都与其规模有关。
问题的规模越⼩,越容易直接求解,解题所需的计算时间也越少。
例如,对于n个元素的排序问题,当n=1时,不需任何计算。
n=2时,只要作⼀次⽐较即可排好序。
n=3时只要作3次⽐较即可,…。
⽽当n较⼤时,问题就不那么容易处理了。
要想直接解决⼀个规模较⼤的问题,有时是相当困难的。
⼆、基本思想及策略分治法的设计思想是:将⼀个难以直接解决的⼤问题,分割成⼀些规模较⼩的相同问题,以便各个击破,分⽽治之。
分治策略是:对于⼀个规模为n的问题,若该问题可以容易地解决(⽐如说规模n较⼩)则直接解决,否则将其分解为k个规模较⼩的⼦问题,这些⼦问题互相独⽴且与原问题形式相同,递归地解这些⼦问题,然后将各⼦问题的解合并得到原问题的解。
这种算法设计策略叫做分治法。
如果原问题可分割成k个⼦问题,1<k≤n,且这些⼦问题都可解并可利⽤这些⼦问题的解求出原问题的解,那么这种分治法就是可⾏的。
由分治法产⽣的⼦问题往往是原问题的较⼩模式,这就为使⽤递归技术提供了⽅便。
在这种情况下,反复应⽤分治⼿段,可以使⼦问题与原问题类型⼀致⽽其规模却不断缩⼩,最终使⼦问题缩⼩到很容易直接求出其解。
这⾃然导致递归过程的产⽣。
分治与递归像⼀对孪⽣兄弟,经常同时应⽤在算法设计之中,并由此产⽣许多⾼效算法。
三、分治法适⽤的情况分治法所能解决的问题⼀般具有以下⼏个特征:1) 该问题的规模缩⼩到⼀定的程度就可以容易地解决2) 该问题可以分解为若⼲个规模较⼩的相同问题,即该问题具有最优⼦结构性质。
分治法 “分”而治之.ppt
当n→∞,S(n) ∝ U(n) ,而U(n) = Θ(logn)
谢谢你的观赏
19
基于二元比较树的分析
若x在A中出现,则算法的执 行过程在一个圆形的内结点处结 束。 若x不在A中出现,则算法的 执行过程在一个方形的外结点处 结束
——外结点不代表元素的比 较,因为比较过程在该外结点的 上一级的内结点处结束。
5
2
7
1 3 68
4
9
谢6 谢你的观赏
谢谢你的观赏
8
分治法的另一种模型表示
proc dividandconquer(n)
if n<=n0 then g(n)
else { divid n into small suninstances
n1 n2 n3…nk
for i=1 to k do
yi=dividandconquer(ni)
return merge(y1…yk) }
效率较好。 一般进行2分法。
谢谢你的观赏
谢谢你的观赏
10
3.2 二分检索(折半查找) 1. 问题的描述
已知一个按非降次序排列的元素表a1, a2, …,an,判定某给定的元素x是否在该表中出 现。
若是,则找出x在表中的位置并返回其所在下标 若非,则返回0值。
谢谢你的观赏
谢谢你的观赏
11
故:成功检索在i级终止所需要的元素比较次数是i 不成功检索在i级外部结点终止的元素比较次数是i-1
谢谢你的观赏
谢谢你的观赏
21
BINSRCH计算复杂度的理论分析
1)不成功检索的最好、最坏和平均情况的计算时间
NOIP基础算法综合---分治与贪心
分析
• B、求方程的所有三个实根
• 所有的根的范围都在-100至100之间,且根与根之差的绝 对值>=1。因此可知:在[-100,-99]、[-99,-98]、……、[99, 100]、[100,100]这201个区间内,每个区间内至多只能 有一个根。即:除区间[100,100]外,其余区间[a,a+1], 只有当f(a)=0或f(a)·f(a+1)<0时,方程在此区间内才有解。 若f(a)=0 ,解即为a;若f(a)·f(a+1)<0 ,则可以利用A中所 述的二分法迅速出找出解。如此可求出方程的所有的解。
while(i<=mid)temp[p++]=a[i++]; while(j<=right)temp[p++]=a[j++]; for(i=left;i<=right;i++)a[i]=temp[i]; }
【变形1】逆序对数目
• 例题:求“逆序对”。 • 给定一整数数组A=(A1,A2,…An), 若i<j且Ai>Aj,
核心参考代码
void divide(double x1,double x2) { double x0,y0,y1,y2;
x0=(x1+x2)/2; y1=cal(x1);y2=cal(x2);y0=cal(x0); if(x2-x1<0.00001&&y1*y2<0)
{printf("%.4f ",(x2+x1)/2);return;} if(y1*y0<0||x0-x1>1) divide(x1,x0); if(y0*y2<0||x2-x0>1) divide(x0,x2); }
分治算法 选择题
分治算法选择题
关于分治算法的选择题较多,比如:
1. 求众数:
问题描述:给定一个大小为n的数组,找到其中的众数。
众数是指在数组中出现次数大于⌊ n/2 ⌋的元素。
你可以假设数组是非空的,并且给定的数组总是存在众数。
示例1:输入:[3,2,3],输出:3
示例2:输入:[2,2,1,1,1,2,2],输出:2
题解:对数组排序,由于众数大于n/2,则n/2+1处一定是该众数。
又由于索引从0开始,即索引n/2为众数。
算法时间复杂度O(nlogn)。
2. 分治法的适用条件是什么?
答案:A.问题可以分解为规模较小的子问题;B.小规模子问题可解;C.子问题可合并为问题的解;D.子问题相互独立。
3. 分治法的设计思想是什么?
答案:A.大事化小,各个击破,分而治之。
4. 每次都将问题分解为原问题规模的一半进行求解,称为什么法?
答案:A.二分法。
5. 分治法一般在每一层递归上有哪三个步骤?
答案:A.分解、解决、合并。
6. 减治法是什么?
答案:A.把一个问题转化成一个子问题来解决,从子问题的解得到原问题的解。
7. 分治法将原问题分解为若干个什么规模的子问题?
答案:A.规模较小、相互独立、完全相同。
如需更多关于分治算法的选择题及答案,可以咨询专业算法人士获取更多资源。
《算法设计与分析》课程实验报告 (分治法(三))
《算法设计与分析》课程实验报告实验序号:04实验项目名称:实验4 分治法(三)一、实验题目1.邮局选址问题问题描述:在一个按照东西和南北方向划分成规整街区的城市里,n个居民点散乱地分布在不同的街区中。
用x 坐标表示东西向,用y坐标表示南北向。
各居民点的位置可以由坐标(x,y)表示。
街区中任意2 点(x1,y1)和(x2,y2)之间的距离可以用数值∣x1−x2∣+∣y1−y2∣度量。
居民们希望在城市中选择建立邮局的最佳位置,使n个居民点到邮局的距离总和最小。
编程任务:给定n 个居民点的位置,编程计算邮局的最佳位置。
2.最大子数组问题问题描述:对给定数组A,寻找A的和最大的非空连续子数组。
3.寻找近似中值问题描述:设A是n个数的序列,如果A中的元素x满足以下条件:小于x的数的个数≥n/4,且大于x的数的个数≥n/4 ,则称x为A的近似中值。
设计算法求出A的一个近似中值。
如果A中不存在近似中值,输出false,否则输出找到的一个近似中值4.循环赛日程表问题描述:设有n=2^k个运动员要进行网球循环赛。
现要设计一个满足以下要求的比赛日程表:每个选手必须与其他n-1个选手各赛一次,每个选手一天只能赛一次,循环赛一共进行n-1天。
二、实验目的(1)进一步理解分治法解决问题的思想及步骤(2)体会分治法解决问题时递归及迭代两种不同程序实现的应用情况之差异(3)熟练掌握分治法的自底向上填表实现(4)将分治法灵活于具体实际问题的解决过程中,重点体会大问题如何分解为子问题及每一个大问题涉及哪些子问题及子问题的表示。
三、实验要求(1)写清算法的设计思想。
(2)用递归或者迭代方法实现你的算法,并分析两种实现的优缺点。
(3)根据你的数据结构设计测试数据,并记录实验结果。
(4)请给出你所设计算法的时间复杂度的分析,如果是递归算法,请写清楚算法执行时间的递推式。
四、实验过程(算法设计思想、源码)1.邮局选址问题(1)算法设计思想根据题目要求,街区中任意2 点(x1,y1)和(x2,y2)之间的距离可以用数值∣x1−x2∣+∣y1−y2∣度量。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
7个乘法和10个加 (减)法
则:
C11 P S T V C12 P T C21 Q S C22 P R Q U
8个加(减)法
斯特拉森时间复杂度
b T (n) 2 7 T ( n / 2) an n2 n2
斯特拉森矩阵乘法目前只具有理论意义。 n2相当大时才会优于通常的矩阵乘法。 T (只有当 n) an (1 7 / 4 (7 / 4) 2 (7 / 4) k 1 ) 7 k T (1) (n≧120 2 ) log n log n
( (i 1)) (n 2 )
1
n
最坏情况时间是O(n)的选择算法
基本思想:精心挑选划分元素v 方法:二次取中间值 目的:使v比一部分元素小比另一部分元素大
使用二次取中规则的选择算法的说明性描述
public static int Select2(int a[],int k,int n) { //在集合a中找第k小元素 ① 若n≤r,则采用插入法直接对a分类并返回第k小元素。 ② 把a分成大小为r的个子集合,忽略剩余的元素。 ③ 设 M m1 , m2 ,, mn / r 是上面 n / r 个子集合的中间值的集合。 ④ v Select 2( M , / 2 n / r ) n / r , ⑤ 用Partition划分a,v作为划分元素。 ⑥ 假设v在位置j。 ⑦ case k=j: return(v);
最坏情况时间
Select的最坏情况时间是O( n 2 ) Select的平均情况时间是O(n)
最坏情况下的特例: 输入a恰好使对Partition的第i次调用选用的划分
元素是第i小元素,而k=n。 此时,(区间下界)m随
着Partition的每一次调用而仅增加1,j保持不变。
Partition最终需要调用n次。 则n次调用的时间总量是:
2) 如何保存n / r 个子集合的中间值 注:各组找到的中间元素值将调整到数组a的前 部,连续保存,从而可用递归调用的方式对这些中间 值进行排序并找出中间值的中间值。
Select2的实现
public static int Sel(int m,int p,int k) {//返回一个i,使得i属于[m,p],且a[i]是a[m:p]中第k小元素,r是一个全程变量,其取值为一 个大于1的整数。 int i,j,n,temp; if(p-m+1<=r) { InsertionSort(m,p); return(m+k-1);//返回第k小元素的位置 } while(true) { n=p-m+1; //元素数 for(i=1;i<=n/r;i++) //计算中间值 { InsertionSort(m+(i-1)*r,m+i*r-1); //将中间值收集到a[m:p]的前部 temp=a[m+i-1]; a[m+i-1]=a[m+(i-1)*r+r/2-1]; a[m+(i-1)*r+r/2-1]=temp; } j=Sel(m,m+n/r-1, ((n/r)+1)/2); //二次取中求mm temp=a[m]; a[m]=a[j]; a[j]=temp; //交换a[m]与a[j] j=Partition(m,p); if((j-m+1)==k) return(j); else if(j-m+1>k) p=j-1; else { k=k-(j-m+1); m=j+1; } } }
k<j: 设S是a[1:j-1]中元素的集合
return(Select2(S,k,j-1)) else:设R是a[j+1:n]中元素的集合 return(Select2(R,k-j,n-j))
}
Select2的待解决问题
算法中需要解决的两个问题 1) 如何确定子集合的中间值 当r较小时,采用InsertionSort(a,i,j)直接对每组 的r个元素分类,在分类好的序列中,中间元素即为当 前r个元素中的中间位置下标对应的元素。
cn (7 / 4)
7
cnlog 4 log 7 log 4 nlog 7 (c 1)nlog 7 (nlog 7 ) (n 2.81 )
斯特拉森矩阵乘法
设矩阵A和B是两个n×n矩阵,讨论矩阵加法的时间复 杂度和矩阵乘法的时间复杂度。 观察:矩阵乘法的花费比矩阵加法大 。 矩阵加法:Θ( n2 )
3 矩阵乘法:Θ( n)
C (i, j )
1k n
A(i, k )B(k , j)
1 i, j n
分治法解决矩阵乘法(降低计算复杂度)
利用Partition实现的选择算法
public static void Select(int n,int k) {//在数组a[1],…,a[n]中找第k小元素s并把它放在位置k,假设1≤k≤n。 //将剩下的元素按如下方式排列,使a[k]=t,对于1≤m<k,有a[m]≤t;对 于k<m≤n,有a[m]≥t。a[n+1]=+∞ int m,r,j; m=1;r=n+1;a[n+1]=10000; while(true) //每当进入这一循环时,1≤m≤k≤r≤n+1 { j=r; //将剩余元素的最大下标加1后给j j=Partition(m,j); //返回j,它使得a[j]是第j小的值 if(k<j) r=j; //j是新的上界 else if(k==j) return; else m=j+1; //j+1是新的下界 } }
斯特拉森矩阵乘法
假设:n=2k
A11 A 21
其中:
A12 B11 B A22 21
B12 C11 C12 B22 C21 C22
技巧:增加加法减少乘法
C11 A11B11 A12 B21 C12 A11 B12 A12 B22 C21 A21 B11 A22 B21 C22 A21 B12 A22 B22
利用Partition实现的选择算法实例分析
算法复Байду номын сангаас度分析
假设 ① a中的元素互异; ② 随机选取划分元素,且选择a[m:p-1]中任一元素作为划分 元素的概率相同。 分析 在Select中每次调用Partition(m,j),所需的元素比较次是 j-m+1。 在执行一次Partition后,或者找到第k小元素,或者将在缩 小的子集(a[m,k-1]或a[k+1,j])中继续查找。缩小的子集的元 素数将至少比上一次划分的元素数少1。
斯特拉森矩阵乘法的思想
令:
P ( A11 A22 )( B11 B22 ) Q ( A21 A22 ) B11 R A11 ( B12 B22 ) S A22 ( B21 B11 ) T ( A11 A12 ) B22 U ( A21 A11 )( B11 B12 ) V ( A12 A22 )( B21 B22 )
选择问题
1. 问题描述 在n个元素的表a[1:n]中确定第k小元素,1≤k≤n。 2. 设计思路 利用Partition过程。在第一次划分后划分元素v测定在a[j] 的位置上,则有j-1个元素小于或等于a[j],且有n-j个元素大 于或等于a[j]。此时, 若k=j,则a[j]即是第k小元素;否则, 若k<j,则a[1:n]中的第k小元素将出现在a[1:j-1]中, 且仍是a[1:j-1]中的第k小元素; 若k>j,则a[1:n]中的第k小元素将出现在a[j+1:n], 是a[j+1:n]中的第k-j小元素。