当前位置:首页 » 操作系统 » 常见的算法题

常见的算法题

发布时间: 2025-05-06 01:48:57

1. 贪心算法的例题分析

例题1、
[0-1背包问题]有一个背包,背包容量是M=150。有7个物品,物品不可以分割成任意大小。
要求尽可能让装入背包中的物品总价值最大,但不能超过总容量。
物品 A B C D E F G
重量 35kg 30kg 6kg 50kg 40kg 10kg 25kg
价值 10$ 40$ 30$ 50$ 35$ 40$ 30$
分析:
目标函数:∑pi最大
约束条件是装入的物品总重量不超过背包容量:∑wi<=M(M=150)
⑴根据贪心的策略,每次挑选价值最大的物品装入背包,得到的结果是否最优?
⑵每次挑选所占重量最小的物品装入是否能得到最优解?
⑶每次选取单位重量价值最大的物品,成为解本题的策略。
值得注意的是,贪心算法并不是完全不可以使用,贪心策略一旦经过证明成立后,它就是一种高效的算法。
贪心算法还是很常见的算法之一,这是由于它简单易行,构造贪心策略不是很困难。
可惜的是,它需要证明后才能真正运用到题目的算法中。
一般来说,贪心算法的证明围绕着:整个问题的最优解一定由在贪心策略中存在的子问题的最优解得来的。
对于例题中的3种贪心策略,都是无法成立(无法被证明)的,解释如下:
⑴贪心策略:选取价值最大者。
反例:
W=30
物品:A B C
重量:28 12 12
价值:30 20 20
根据策略,首先选取物品A,接下来就无法再选取了,可是,选取B、C则更好。
⑵贪心策略:选取重量最小。它的反例与第一种策略的反例差不多。
⑶贪心策略:选取单位重量价值最大的物品。
反例:
W=30
物品:A B C
重量:28 20 10
价值:28 20 10
根据策略,三种物品单位重量价值一样,程序无法依据现有策略作出判断,如果选择A,则答案错误。
【注意:如果物品可以分割为任意大小,那么策略3可得最优解】
对于选取单位重量价值最大的物品这个策略,可以再加一条优化的规则:对于单位重量价值一样的,则优先选择重量小的!这样,上面的反例就解决了。
但是,如果题目是如下所示,这个策略就也不行了。
W=40
物品:A B C
重量:25 20 15
价值:25 20 15
附:本题是个DP问题,用贪心法并不一定可以求得最优解,以后了解了动态规划算法后本题就有了新的解法。
例题2、
马踏棋盘的贪心算法
123041-23 XX
【问题描述】
马的遍历问题。在8×8方格的棋盘上,从任意指定方格出发,为马寻找一条走遍棋盘每一格并且只经过一次的一条路径。
【初步设计】
首先这是一个搜索问题,运用深度优先搜索进行求解。算法如下:
⒈ 输入初始位置坐标x,y;
⒉ 步骤 c:
如果c> 64输出一个解,返回上一步骤c--
(x,y) ← c
计算(x,y)的八个方位的子结点,选出那些可行的子结点
循环遍历所有可行子结点,步骤c++重复2
显然⑵是一个递归调用的过程,大致如下:
C++程序: #defineN8voiddfs(intx,inty,intcount){inti,tx,ty;if(count>N*N){output_solution();//输出一个解return;}for(i=0;i<8;i++){tx=hn[i].x;//hn[]保存八个方位子结点ty=hn[i].y;s[tx][ty]=count;dfs(tx,ty,count+1);//递归调用s[tx][ty]=0;}}Pascal程序: ProgramYS;ConstFXx:array[1..8]of-2..2=(1,2,2,1,-1,-2,-2,-1);FXy:array[1..8]of-2..2=(2,1,-1,-2,-2,-1,1,2);VarRoad:array[1..10,1..10]ofinteger;x,y,x1,y1,total:integer;ProcereFind(x,y:integer);varNx,Ny,i:integer;BeginFori:=1to8dobegin{8个方向}If(x+FXx[i]in[1..8])and(y+FXy[i]in[1..8])Then{确定新坐标是否越界}IfRoad[x+Fxx[i],y+Fxy[i]]=0Thenbegin{判断是否走过}Nx:=x+FXx[i];Ny:=y+FXy[i];Road[Nx,Ny]:=1;{建立新坐标}If(Nx=x1)and(Ny=y1)Theninc(total)elseFind(Nx,Ny);{递归}Road[Nx,Ny]:=0{回朔}endendEnd;BEGIN{Main}Total:=0;FillChar(Road,sizeof(road),0);Readln(x,y);{读入开始坐标}Readln(x1,y1);{读入结束坐标}If(x>10)or(y>10)or(x1>10)or(y1>10)Thenwriteln('Error'){判断是否越界}ElseFind(x,y);Writeln('Total:',total){打出总数}END.这样做是完全可行的,它输入的是全部解,但是马遍历当8×8时解是非常之多的,用天文数字形容也不为过,这样一来求解的过程就非常慢,并且出一个解也非常慢。
怎么才能快速地得到部分解呢?
【贪心算法】
其实马踏棋盘的问题很早就有人提出,且早在1823年,J.C.Warnsdorff就提出了一个有名的算法。在每个结点对其子结点进行选取时,优先选择‘出口’最小的进行搜索,‘出口’的意思是在这些子结点中它们的可行子结点的个数,也就是‘孙子’结点越少的越优先跳,为什么要这样选取,这是一种局部调整最优的做法,如果优先选择出口多的子结点,那出口少的子结点就会越来越多,很可能出现‘死’结点(顾名思义就是没有出口又没有跳过的结点),这样对下面的搜索纯粹是徒劳,这样会浪费很多无用的时间,反过来如果每次都优先选择出口少的结点跳,那出口少的结点就会越来越少,这样跳成功的机会就更大一些。这种算法称为为贪心算法,也叫贪婪算法或启发式算法,它对整个求解过程的局部做最优调整,它只适用于求较优解或者部分解,而不能求最优解。这样的调整方法叫贪心策略,至于什么问题需要什么样的贪心策略是不确定的,具体问题具体分析。实验可以证明马遍历问题在运用到了上面的贪心策略之后求解速率有非常明显的提高,如果只要求出一个解甚至不用回溯就可以完成,因为在这个算法提出的时候世界上还没有计算机,这种方法完全可以用手工求出解来,其效率可想而知。

2. 算法面试题——动态规划Kadane’s algorithm

动态规划是大公司面试必考的算法题。其中,“最大子序和”是其中的经典:

面试官:年轻人,前面表现的不错嘛!看下这道题,给你这个数组[-2, 1, -3, 4, -1, 2, 1, -5, 4],找到其中和最大的连续子数组,返回最大值。

我:

使用Kadane’s Algorithm解决这个问题。Kadane’s Algorithm可以简单理解为动态规划的一种应用,它通过维护两个变量:localMax和globalMax,来计算连续子数组的最大和。localMax表示以当前元素结尾的子数组的最大和,而globalMax则记录遍历过程中发现的最大值。

我们以数组[-2, 1, -3, 4, -1, 2, 1, -5, 4]为例,从左到右遍历数组。

当遍历到元素4时,localMax为4(因为4是当前元素),globalMax也更新为4(因为这是遍历到的第一个元素)。

遍历到元素-1时,我们考虑两种情况:
1. 包括前面的元素4,即localMax更新为4 + (-1) = 3。
2. 不包括前面的元素4,即从元素-1开始新的子数组。由于-1 > 3,我们选择-1作为新的localMax。
因此,globalMax更新为-1。

遍历到元素2时,localMax为2(因为2 > 1),globalMax更新为2。

遍历到元素1时,localMax更新为3(因为3 > 2 + 1),globalMax也更新为3。

遍历到元素-5时,我们考虑两种情况:
1. 包括前面的元素3,即localMax更新为3 + (-5) = -2。
2. 不包括前面的元素3,即从元素-5开始新的子数组。由于-5 < 3,我们选择3作为新的localMax。
因此,globalMax更新为3。

遍历到元素4时,localMax为4(因为4 > 3 + 4),globalMax更新为4。

最终,我们得到数组的最大子序和为4。

动态规划和Kadane’s Algorithm在解决“最大子序和”问题时,通过维护局部最优解和全局最优解,避免了暴力算法中的重复计算,大大提高了效率。

理解Kadane’s Algorithm的关键在于,它通过比较当前元素和当前元素加上前面元素的最大和,来决定是否更新局部最优解。这样,我们不仅避免了重复计算,还能在遍历过程中不断更新全局最优解。

此外,Kadane’s Algorithm在解决类似问题时,如“买卖股票的最佳时机”,同样可以应用动态规划的思想,通过维护局部最优解和全局最优解,来找到最优策略。

动态规划和Kadane’s Algorithm是解决一系列动态规划问题的有效工具,通过理解和应用这些算法,可以大大提高解决问题的效率和准确性。

热点内容
编程小博客 发布:2025-05-06 06:34:00 浏览:925
php循环html 发布:2025-05-06 06:31:16 浏览:808
手机电话号如何设置成服务器错误 发布:2025-05-06 06:26:54 浏览:573
电脑怎么弄共享服务器 发布:2025-05-06 06:26:23 浏览:882
linuxfirefox安装 发布:2025-05-06 06:14:23 浏览:685
java下载服务器文件 发布:2025-05-06 06:09:21 浏览:626
cpu配置过低是什么意思 发布:2025-05-06 06:09:13 浏览:640
数据库字段命名 发布:2025-05-06 05:59:19 浏览:36
页面源码下载 发布:2025-05-06 05:51:29 浏览:134
勇士食堂安卓怎么登录 发布:2025-05-06 05:49:19 浏览:531