问题描述

小Hi和小Ho在经历了螃蟹先生的任务之后被奖励了一次出国旅游的机会,于是他们来到了大洋彼岸的美国。美国人民的生活非常有意思,经常会有形形色色、奇奇怪怪的活动举办,这不,小Hi和小Ho刚刚下飞机,就赶上了当地的迷宫节活动。迷宫节里展览出来的迷宫都特别的有意思,但是小Ho却相中了一个其实并不怎么像迷宫的迷宫——因为这个迷宫的奖励非常丰富~

于是小Ho找到了小Hi,让小Hi帮助他获取尽可能多的奖品,小Hi把手一伸道:“迷宫的介绍拿来!”

小Ho选择的迷宫是一个被称为“数字三角形”的n(n不超过200)层迷宫,这个迷宫的第i层有i个房间,分别编号为1..i。除去最后一层的房间,每一个房间都会有一些通往下一层的房间的楼梯,用符号来表示的话,就是从第i层的编号为j的房间出发会有两条路,一条通向第i+1层的编号为j的房间,另一条会通向第i+1层的编号为j+1的房间,而最后一层的所有房间都只有一条离开迷宫的道路。这样的道路都是单向的,也就是说当沿着这些道路前往下一层的房间或者离开迷宫之后,小Ho没有办法再次回到这个房间。迷宫里同时只会有一个参与者,而在每个参与者进入这个迷宫的时候,每个房间里都会生成一定数量的奖券,这些奖券可以在通过迷宫之后兑换各种奖品。小Ho的起点在第1层的编号为1的房间,现在小Ho悄悄向其他参与者弄清楚了每个房间里的奖券数量,希望小Hi帮他计算出他最多能获得多少奖券。

提示一:盲目贪心不可取,搜索计算太耗时

提示二:记忆深搜逞神威,宽度优先解难题

提示三:总结归纳提公式,减少冗余是真理

提示一:盲目贪心不可取,搜索计算太耗时

小Hi拿到迷宫的介绍,仔细想了想,问道:“你自己有什么思路么?”

小Ho想了想,道:“反正每次只有两种选择,我就选通向有更多奖券的房间的那条路呗~”

小Hi笑了笑:“你这叫盲目贪心,如果我是迷宫的创造者,一定在第2层的第1个房间放10张奖券,第2个房间一张不放,然后在第3层的第3个房间放999张奖券,这样你就因为占一点小便宜就失掉了大头,你说是不是?”

小Ho一惊:“似乎是的!切记不可因小失大,那……既然我们都是学计算机的学生,不如我们用计算机枚举一下所有可能的路径,然后找到最优的那一条?”

小Hi点了点头:“自然是要用计算机来解决这个问题,不过你得先计算一下总共有多少条可能的路径,估算一下运行时间,这样才能确定能不能在迷宫节结束之前计算完是不是~”

“我来算算,从出发点开始,每次我都有两种选择,而我总共要做n-1次选择,也就是2的n-1次方,而n最大可能是200,也就是说有2^199条路径……哪怕每条路径我只用一次运算就可以算出它的奖券总数,我毕业之前都肯定算不出来了吧QAQ”小Ho得出了一个非常忧伤的结论。

“是的,但是这种搜索的方法其实是可以有方法进行优化的哦~”小Hi循循诱导着。

“什么样的方法?”

“别急,听我慢慢说~

提示二:记忆深搜逞神威,宽度优先解难题

“我们可以来分分情况,我们要搜索所有的路径,然后依次计算每条路径的收益,这是我们现在的问题模型是不是?”小Hi问道。

小Ho点头:“仔细分析这个迷宫的话,我们有两种搜索方法,一种是深度优先搜索,就像这幅图画的一样,我们先试着顺着每个房间的第一条路一直下去,一直走到最后一层,然后返回至倒数第二层,选择第二条路走到最后一个房间,然后返回到倒数第三层,选择第二条路走到倒数第二层,然后选择第一条路走到最后一层……期间维护已经走过的房间的奖券之和sum,然后每次到达最后一层的房间的时候将sum与全局最优解ans进行比较,如果sum>ans的话就用sum替换ans。这样在所有路径都计算过一遍之后,Ans中存储的就是我们要的答案了~~而第二种……”

“先别说第二种,我们来看看这种方法有什么办法优化么~”小Hi打断了小Ho的喋喋不休。

“哦,但是该怎么优化呢?”小Ho问道。

“老规矩,我们来看看这个过程中有什么冗余计算~”小Hi还是那一套说辞,也不担心小Ho听了这么多遍听厌:“看这张图,当你枚举到绿色的‘-》右-》左’这样一条路径的时候,是不是发现它和红色的‘-》左-》右’这条路径一样都到达了第三层的第二个房间?”

“是的!”

“在你枚举到绿色路径的时候,是不是红色路径已经被枚举过了?”小Hi接着问道。

“没错!“

“那么你看,绿色路径的奖券和是8,红色路径的奖券和是10,而你们都处于了相同的结点——第3层的第2个房间上,这时候无论你绿色路径接下来延续出什么样的路径,我将从第3层第2个房间开始的这一段连在红色路径之后的话,奖券和都会要比绿色路径要多?”小Hi继续问。

“嗯……对的~”

“而这些路径中的最优值都肯定不超过Ans,不然Ans就会在当时变得和它们一样,这不就说明了绿色路径无论接下来如何走,都不可能对Ans造成任何变化,那我们是不是完全可以不进行接下来的计算了呢?”小Hi做出了最后的结论。

“是的……为了进行这样的优化,我们需要记录一个值best(i, j)——当前搜索过的路径中到达第i层第j个房间时最多能获取多少奖券,然后在每次进入一个房间的时候,都检查当前的sum与best(i, j)的大小关系,如果sum小于等于best(i, j)的话,就没有必要继续搜索了呢。比如在之前的例子中,当通过绿色路径到达第3层第2个房间的时候,best(i, j)=10,而sum = 8,所以是没有必要继续往下搜索的。”聪明的小Ho很快就将这个算法总结了出来。

“而这种做法,由于是通过之前的记忆来避免不必要的搜索,所以被大家称为‘记忆化搜索’”小Ho笑道:“这样,只要不是那种非常坏的卡你的情况——比如第i层第j个房间的奖券数是j这种,都能够在O(n^2)的复杂度通过了呢。”

“而如果我先右再左呢?”小Ho不死心。

“那我就第i层第j个房间的奖券数是i-j”小Ho邪恶的笑着:“其实你不用纠结这个,你只要随机一下就能够保证几乎所有情况都是O(n^2)的时间复杂度,或者说平均复杂度是O(n^2),但是你的最坏情况复杂度肯定是降不下去的,所以还是老老实实想想别的方法吧~比如之前你想说的第二种方法?”

“与深度优先搜索相对应的自然就是宽度优先遍历啦~如果我们用<i, j, k>表示到达第i行的第j个房间,当前获得的奖券数为k这样一个状态,rewards(i, j)表示第i层第j个房间中的奖券数,那么遍历方式就是利用一个队列,先将ə, 1, rewards(1, 1)>这个状态放入队列中。然后每次取出队首<i, j, k>,如果i==n,就用k更新Ans,不然就将它代表的房间所能到达的两个房间的状态<i + 1, j, k + rewards(i + 1, j)>和<i + 1, j + 1, k + rewards(i + 1, j + 1)>放入队尾,直到整个队列清空,这时候Ans就是我们要的答案。”小Ho缓缓道来。

“那你觉得这个地方又有什么可以优化的呢?”

“本来我还没想到,但是你说了之前深度优先搜索的问题之后我就有点思路了,你看状态(2, 1, 8)和状态(2, 2, 6)在处理时一个会拓展出(3, 2, 10)这个状态,而另一个会拓展出(3, 2, 8)这个状态,但是就像我们之前的分析那样(3, 2, 8)这个状态是没有什么用的,它之后无论怎么走,换成(3, 2, 10)按它一样的走法肯定获得的奖券数要更多。”小Ho触类旁通,一下就找到了问题所在:“所以我可以像深度优先搜索那样用一个best(i, j)记录当前搜索过的路径中到达第i层第j个房间时最多能获取多少奖券,然后在一个状态(i, j, k)想要入队的时候,判断k与best(i, j)的大小,如果k小于等于best(i, j)就没有必要进队了~”

“看来你还陷在了死胡同里,你这样在最坏情况复杂度根本没有变化不是?”小Hi无奈道:“你仔细想想,按照宽度优先搜索的顺序,是不是一个状态(i, j, k)在拓展出(i', j', k')的时候,(i', j', k')要么从来没有入队列,要么就仍然在队列里?”

“唔...这个问题看来,基本上就是这样一层一层的顺序出队列的,那还是那个问题,如果我在判断状态(i, j, k)是否应该入队列的时候,发现了k大于best(i, j),这个时候如果我再去判断一下队列中是否有一个(i, j, k')的状态。如果没有,就将(i, j, k)入队列,不然因为k'大于k,那么就像我们之前分析的那样(i, j, k')这个状态时无用的,不如就用k去替换掉状态(i, j, k')中的k',这样也会减少运算不是?”

“没错~而且你仔细分析的话,每个房间都只会入队列和出队列一次,这样就保证了平均情况时间复杂度和最坏情况时间复杂度都是O(n^2)哦!~”

“原来是这样,那么现在问题差不多解决了呢!”小Ho高兴道:“赶紧算出来,我好去拿奖品。”

“急什么!你从这之中可学到了什么知识么?”小Hi拦住了急冲冲的小Ho,问道。

“啊?”

提示三:总结归纳提公式,减少冗余是真理

“且不说深度优先这样一个最坏情况下仍然有问题的方法,我们来说这个宽度优先搜索,我们能够使用best(i, j)来进行优化,无非是因为这个问题存在这样两个性质。”小Hi道。

“第一便是无论我是通过怎么样的方式到达第i层第j个房间的,我之前做出的选择不会对我之后的选择做出限制与优待。就像如果我规定至少要向右走3次,那么状态就不仅仅是(i, j, k)这样,还要加上一个已经向右走的次数t,那么你觉得还能就直接像我们之前的方法进行计算么?如果到达第i层第j个房间的路上向右走过3次了,那么之后的走法就没有任何限制,不然就仍会有一个至少要向右走一定次数的限制。这样对于两个状态(i, j, k, l)和(i, j, k', l'),我们就不能够直接判断出那个状态是更加好的。

“没错~其实就是best(i, j)有两个关键字进行参考,而这样的话就不是任意两个状态都可以进行比较的了。”小Ho总结道。

“这样的性质被我们称为无后效性,顾名思义,当前的抉择不会对后面抉择产生影响。”

“那第二个性质呢?”

“第二个性质被我们称为重复子问题,在我们之前的例子中,我们将从起点出发到走出迷宫的最优路分解成了两个子问题,其一是从第2层的第1个房间走出迷宫的最优路,其二是从第2层的第2个房间走出迷宫的最优路,只要能算出这两个部分的最优值,我就可以知道从起点出发到走出迷宫的最优路。”小Hi道:”而按照这样的方法,这两个子问题都有一个相同的子问题——从第3层的第2个房间出发走出迷宫的最优路。”

“这就是重复的子问题了,而我们的做法就是,重复的子问题只计算一次!也就是说我们要先计算出从起点到达第3层的第2个房间的最优线路,然后再考虑接下来怎么走~”小Ho感觉自己渐渐的把握到了关键之处。

“对的,所以综合这样两个性质,我们可以考虑用best(i, j)表示从起点出发,走到第i层第j个房间出发最多可以获得的奖券数。”小Hi提出了一个新的定义:“那么我们要求的答案,便是所有best(n, j)中最大的那一个对不对?

“是的,那该如何求这个呢……”小Ho思索着:“第i层第j个房间可以由第i-1层第j个房间和第i-1层第j-1个房间到达,所以best(i, j) = max{best(i - 1, j - 1), best(i - 1, j)} + rewards(i, j)这样么?”

“是的,这个公式便是我们俗称的状态转移方程,你要做的,就是按照这样的顺序,依次计算出每一个房间的best(i, j),然后在最后统计所有best(n, j)的最大值,就可以得到答案了。”

“这个过程和宽度优先搜索的确很像,但是由于顺序是确定了的,我可以用一个两重循环,外层是i从1到n枚举层数,内层是j从1到i枚举房间编号,就可以直接进行计算了!”小Ho道。

“你这个公式的边界情况还有待考虑嘛,比如j=1的时候就不需要取max了,直接使用best(i - 1, j)计算即可~”小Hi指出了小Ho存在的一个问题。

“哦~那么j=i的时候也有相似的处理是吧。”

“是的呢!所以快去写程序吧~”小Hi笑了笑:“抓紧时间,迷宫节就要结束了哦!”

输入

每个测试点(输入文件)有且仅有一组测试数据。

每组测试数据的第一行为一个正整数n,表示这个迷宫的层数。

接下来的n行描述这个迷宫中每个房间的奖券数,其中第i行的第j个数代表着迷宫第i层的编号为j的房间中的奖券数量。

测试数据保证,有100%的数据满足n不超过100

对于100%的数据,迷宫的层数n不超过100

对于100%的数据,每个房间中的奖券数不超过1000

对于50%的数据,迷宫的层数不超过15(小Ho表示2^15才3万多呢,也就是说……)

对于10%的数据,迷宫的层数不超过1(小Hi很好奇你的边界情况处理的如何?~)

对于10%的数据,迷宫的构造满足:对于90%以上的结点,左边道路通向的房间中的奖券数比右边道路通向的房间中的奖券数要多。

对于10%的数据,迷宫的构造满足:对于90%以上的结点,左边道路通向的房间中的奖券数比右边道路通向的房间中的奖券数要少。

输出

对于每组测试数据,输出一个整数Ans,表示小Ho可以获得的最多奖券数。

Sample Input

5
2
6 4
1 2 8
4 0 9 6
6 5 5 3 6

Sample Output

28
 /*
问题 第i行有i个数字的数字三角形,从第一行走到末行,每个数字只能向直下和下右走,计算并输出最多奖券数
变量 行数,房间数,奖券数
抽象问题 从起点位置走到第i行第j个房间,最多的奖券数价值是多少
无后效性 之前作出的选择对后面的选择没有限制和优待
重复子问题 将从起点出发到走出迷宫的最优路分解成了两个子问题,
其一是从第2层的第1个房间走出迷宫的最优路,
其二是从第2层的第2个房间走出迷宫的最优路,只要能算出这两个部分的最优值,我们就可以知道从起点出发到走出迷宫的最优路。
照这样的方法,这两个子问题都有一个相同的子问题——从第3层的第2个房间出发走出迷宫的最优路。 状态 best[i][j]表示从起点出发到达第i行第j个房间,能够获得的最多奖券数
状态转移 best[i][j]=max(best[i-1][j],best[i-1][j-1]) + map[i][j];
边界确定 j=1时,best[i][j]=best[i-1][j] + map[i][j];
j=i时,best[i][j]=best[i-1][j-1] + map[i][j];
*/
#include<stdio.h>
/*const int N=110;GCC编译器中const定义的N会报错,因为在C语言中,const不是一个真真正正的常量,其代表的含义仅仅是只读。
使用const声明的对象是一个运行时对象,无法使用其作为某个量的初值、数组的长度、case的值或在类型的情形中使用。需要替换成
下面的define定义*/
#define N 100
int map[N][N];
int best[N][N];
int max(int x,int y){
return x>y?x:y;
}
int main()
{
int n,i,j;
while(scanf("%d",&n) != EOF)
{
for(i=;i<=n;i++)
for(j=;j<=i;j++)
scanf("%d",&map[i][j]);
best[][]=map[][];
for(i=;i<=n;i++){
for(j=;j<=i;j++){
if(j==)
best[i][j]=best[i-][j]+map[i][j];
else if(j==i)
best[i][j]=best[i-][j-]+map[i][j];
else
best[i][j]=max(best[i-][j],best[i-][j-]) + map[i][j];
}
}
int ans=-;
for(i=;i<=n;i++){
if(best[n][i] > ans)
ans=best[n][i];
}
printf("%d\n",ans);
}
return ;
}

数字三角形 (简单DP)的更多相关文章

  1. 【递归】数字三角形 简单dp

    [递归]数字三角形 题目描述 对于大多数人来说,“我们是这么的正常,因此也就这么的平庸.”而天才总是与众不同的,所以当邪狼问修罗王:“老大,你蹲在那儿一动不动看了有半个小时了,蚂蚁有那么好看吗?” 修 ...

  2. 算法训练 数字三角形(DP)

    问题描述 (图3.1-1)示出了一个数字三角形. 请编一个程序计算从顶至底的某处的一条路 径,使该路径所经过的数字的总和最大. ●每一步可沿左斜线向下或右斜线向下走: ●1<三角形行数≤100: ...

  3. hihoCoder#1037 : 数字三角形(DP)

    [题目链接]:click here~~ 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 问题描写叙述 小Hi和小Ho在经历了螃蟹先生的任务之后被奖励了一次出国旅游的机会,于是他 ...

  4. dp递推 数字三角形,dp初学者概念总结

    数字三角形(POJ1163)          在上面的数字三角形中寻找一条从顶部到底边的路径,使得路径上所经过的数字之和最大.路径上的每一步都只能往左下或 右下走.只需要求出这个最大和即可,不必给出 ...

  5. 数字三角形 (DP入门)

    7 3     8 8     1     0 2     7     4     4  4     5     2     6     5 给出一个数字三角形.从三角形的顶部到底部有很多条不同的路径 ...

  6. 洛谷P1216 数字三角形【dp】

    题目:https://www.luogu.org/problemnew/show/P1216 题意: 给定一个三角形.从顶走到底,问路径上的数字之和最大是多少. 走的时候可以往左下(实际上纵坐标不变) ...

  7. codevs——T1220 数字三角形

    http://codevs.cn/problem/1043/  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 黄金 Gold 题解  查看运行结果     题目描述 Descr ...

  8. 简单DP入门(一) 数字三角形

    数字三角形

  9. 【线性DP】数字三角形

    题目链接 原题链接 题目描述 给定一个如下图所示的数字三角形,从顶部出发,在每一结点可以选择移动至其左下方的结点或移动至其右下方的结点,一直走到底层,要求找出一条路径,使路径上的数字的和最大. 7 3 ...

随机推荐

  1. day06_雷神_面向对象初识

    day_06 递归函数 自己用自己.一般递归100多次,都没有解决的问题,放弃递归. count = 0 def func1(): global count count += 1 print(coun ...

  2. 用指定的用户名和密码无法登录到该ftp服务器

    今天在win2008 R2 服务器上默认部署FTP站点时遇到了两个小问题,在网上找了好久资料后发现还是解决不了问题,最终找到问题的原因,在此共享给大家 1.Windows无法访问此文件夹.请确保输入的 ...

  3. HSmartWindowControl 之 摄像头实时显示( 使用 WPF )

    1.添加Halcon控件,创建WPF项目 在VS2013中创建一个WPF工程,然后添加halcon的控件和工具包,参见: HSmartWindowControl之安装篇 (Visual Studio ...

  4. .net core 与ELK(1)安装Elasticsearch

    1.安装java jdk [elsearch@localhost bin]$ java -version openjdk version "1.8.0_181" OpenJDK R ...

  5. 静态工厂 + DbHelper

    在 .NET 的学习过程中用过三种数据库:Sql Server.Access.SQLite.Sql Server 用得相对多一点点,但是,麻烦,每次用它都需要开服务,而且还费资源,更麻烦的是拷贝到一台 ...

  6. Java计数器之CountDownLatch、CyclicBarrier、Semaphore

    在Java里面有几种可以用于控制线程状态的方法,如CountDownLatch计数器.CyclicBarrier循环栅栏.Sempahore信号量.下面就分别演示下他们的使用方法: CountDown ...

  7. zookeeper客户端命令详解

    今天同事突然向看一下zookeeper中都创建了哪些节点,而我本人对zookeeper的客服端命令了解的很少,有些操作竟然不知道怎么用,于是乎就索性整理一下zookeeper客服端命令的使用,并再此记 ...

  8. flex定位下overflow失效的问题研究

    概述 这是我在写移动端页面遇到的问题及解决方法,记录下来供以后开发时参考,相信对其他人也有用. 问题 之前写移动端页面,有一个顶条是导航条,需要固定在页面顶部,并且里面的元素需要可以左右滚动. 但是当 ...

  9. Vue+WebSocket+ES6+Canvas 制作「你画我猜」小游戏

    Vue+WebSocket+ES6+Canvas 制作「你画我猜」小游戏 转载 来源:jrainlau 链接:https://segmentfault.com/a/1190000005804860 项 ...

  10. java环境的配置与安装(windows、macos、linux)

    一.windows下: 1.下载jdk:https://www.oracle.com2.安装教程:https://www.cnblogs.com/zlslch/p/5658399.html 二.mac ...