搜索+简单dp
前言:即使是简单的递归,在复杂度过高时也可以使用简单的dp。
一般有两种情况,一是利用dp思想求最优子结构进行搜索剪枝,二是利用搜索进行dp数组的填充。
例题一、hdu1978
题目大意:这是一个简单的生存游戏,你控制一个机器人从一个棋盘的起始点(1,1)走到棋盘的终点(n,m)。游戏的规则描述如下:
1.机器人一开始在棋盘的起始点并有起始点所标有的能量。
2.机器人只能向右或者向下走,并且每走一步消耗一单位能量。
3.机器人不能在原地停留。
4.当机器人选择了一条可行路径后,当他走到这条路径的终点时,他将只有终点所标记的能量。
如上图,机器人一开始在(1,1)点,并拥有4单位能量,蓝色方块表示他所能到达的点,如果他在这次路径选择中选择的终点是(2,4)
点,当他到达(2,4)点时将拥有1单位的能量,并开始下一次路径选择,直到到达(6,6)点。
我们的问题是机器人有多少种方式从起点走到终点。这可能是一个很大的数,输出的结果对10000取模。
一开始的思路:既然已知迷宫起点终点,果断dfs。
dfs最重要的一点就是明确函数中应有的参数和结束条件
在这里,很显然终止条件是到达右下角的点,所以函数中参数要有x,y坐标。又因为在每个点有对应的能量,由能量决定下一步能走到哪里进行枚举,所以参数还应该包含当前点的能量。
#include <bits/stdc++.h>
using namespace std;
int n,m,a[109][109];
int ans;//答案
bool p(int q,int w)
{
if(q>=0&&q<n&&w>=0&&w<m)
return true;
else
return false;
}
void dfs(int power,int x,int y)
{
if(x==n-1&&y==m-1)
{
ans++;
ans%=10000;
return;
}
for(int i=0;i<=power;i++)
{
if(x+i>=n)
break;
for(int j=0;j<=power-i;j++)
{
if(y+j>=m)
break;
if(i==0&&j==0)
continue;
dfs(a[x+i][y+j],x+i,y+j);
}
}
}
int main()
{
int t;
cin>>t;
while(t--)
{
cin>>n>>m;
for(int i=0;i<n;i++)
{
for(int j=0;j<m;j++)
cin>>a[i][j];
}
dfs(a[0][0],0,0);
cout<<ans;
}
}
很遗憾,完全超时。想来也是,100*100的棋盘还有能量,怎么可能跑得了,仔细一想剪枝也无从下手,因为每一步都是必须要走的。
但是,我们从终点开始看,很显然最后一行的元素只能一直向右走。那我们从最后一行m-1个元素看起,从这里出发只有一种走法,再看第m-2个元素,如果能量大于1,则可以一步走到终点,也可以先走到m-1,那走法就加上dp[n-1][m-1]。
由此一来,状态转移方程就容易得出。使用dfs从最后一行枚举每个点进行dfs,看它能到达哪些点,每到一个点就加上那个点在dp数组中的值,一步一步向前搜索。
蓝色块是起点能到达的格子,走法就是每个格子的dp数和嘛!
代码如下
#include <bits/stdc++.h>
using namespace std;
int n,m,a[209][209],dp[209][209];
int dfs(int power,int x,int y,int k)
{
int sumn=k;
if(dp[x][y]!=0)
return dp[x][y];
for(int i=0;i<=power;i++)
{
if(x+i>=n)//防止越界
break;
for(int j=0;j<=power-i;j++)
{
if(y+j>=m)//防止越界
break;
if(i==0&&j==0)//不能待在原地不懂
continue;
sumn+=dfs(a[x+i][y+j],x+i,y+j,0);
}
}
return sumn;
}
int main()
{
int t;
cin>>t;
while(t--)
{
cin>>n>>m;
memset(dp,0,sizeof(dp));
for(int i=0;i<n;i++)
{
for(int j=0;j<m;j++)
cin>>a[i][j];
}
dp[n-1][m-1]=1;dp[n-1][m]=dp[n][m-1]=1;。。初始化
for(int i=n-1;i>=0;i--)
for(int j=m-1;j>=0;j--)
dp[i][j]=dfs(a[i][j],i,j,0)%10000;//dfs就是求当前格子的走法
cout<<dp[0][0]<<endl;
}
}
搜索+简单dp的更多相关文章
- hdu2067 简单dp或者记忆化搜索
题意: 小兔的棋盘 Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Sub ...
- 简单dp总结
### 简单dp总结 本文是阅读<挑战程序设计第二版>其中关于dp章节所作总结.将简要描述dp的部分知识. 一.dp是什么? dp在计算机专业学科中全称是动态规划(dynamic prog ...
- HDU 1087 简单dp,求递增子序列使和最大
Super Jumping! Jumping! Jumping! Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 ...
- Codeforces Round #260 (Div. 1) A. Boredom (简单dp)
题目链接:http://codeforces.com/problemset/problem/455/A 给你n个数,要是其中取一个大小为x的数,那x+1和x-1都不能取了,问你最后取完最大的和是多少. ...
- codeforces Gym 100500H A. Potion of Immortality 简单DP
Problem H. ICPC QuestTime Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/gym/100500/a ...
- 简单dp --- HDU1248寒冰王座
题目链接 这道题也是简单dp里面的一种经典类型,递推式就是dp[i] = min(dp[i-150], dp[i-200], dp[i-350]) 代码如下: #include<iostream ...
- poj2385 简单DP
J - 简单dp Crawling in process... Crawling failed Time Limit:1000MS Memory Limit:65536KB 64bit ...
- hdu1087 简单DP
I - 简单dp 例题扩展 Crawling in process... Crawling failed Time Limit:1000MS Memory Limit:32768KB ...
- poj 1157 LITTLE SHOP_简单dp
题意:给你n种花,m个盆,花盆是有顺序的,每种花只能插一个花盘i,下一种花的只能插i<j的花盘,现在给出价值,求最大价值 简单dp #include <iostream> #incl ...
随机推荐
- AJ学IOS 之微博项目实战(7)程序启动新特性用UICollectionViewController实现
AJ分享,必须精品 一:效果 这里实现了大多数app都会有的软件新特性的功能,用的是UICollectionViewController实现的 二:思路 这里用了UICollectionViewCon ...
- redis 浅谈事务
写在前面的话 之前在某个网站上看到一个问题:redis在什么情况下出现事务不会滚的情况,以此为由并结合redis官方文档整理这边笔记.不足之处,请指出,谢谢. 事务 redis支持事务,提供两条重要的 ...
- OSI 七层模型以及TCP/IP模型
OSI 七层模型 定义 OSI(Open System Interconnection)即开放式系统互联通信参考模型.该模型是国际标准化组织(ISO)制定的一个用于计算机或通信系统间互联的标准体系,一 ...
- stand up meeting 1--11
今天国庆同学回中科大考试因此缺席了今天的daily scrum.不过国庆的任务已经基本完成,不会影响项目进度. 今日更新: 分享功能已经完成一个版本,如下图为分享至邮件: 针对AP返回结果中没有Wor ...
- Salesforce LWC学习(十六) Validity 在form中的使用浅谈
本篇参考: https://developer.salesforce.com/docs/component-library/bundle/lightning-input/documentation h ...
- 详解JS闭包概念
闭包理解 1. 如何产生闭包? *当一个嵌套的内部(子)函数引用了嵌套的外部(父)函数的变量(函数)时,产生闭包 2. 闭包到底是什么? * 使用Chrome调试查看 * 理解一 ...
- Springboot:员工管理之查询员工列表(十(6))
构建员工controller com\springboot\controller\EmployeeController.java package com.springboot.controller; ...
- tensor求和( tensor.sum())
1. torch.sum(input, dim, out=None) 参数说明: input:输入的tensor矩阵. dim:求和的方向.若input为2维tensor矩阵,dim=0,对列求和:d ...
- Async,Await 深入源码解析
1.同步与异步 假设存在 IO事件A:请求网络资源 (完成耗时5s) IO事件B:查询数据库 (完成耗时5s) 情况一:线程1工人在发起A请求后,一直阻塞等待,在A响应返回结果后再接着处理事件B,那总 ...
- CG-CTF(3)
CG-CTF https://cgctf.nuptsast.com/challenges#Web 续上~ 第十四题:GBK Injection 是一道注入题: 分析:题目提示了GBK,GBK是一种多字 ...