codevs1169传纸条 不相交路径取最大,四维转三维DP
这个题一个耿直的思路肯定是先模拟。。
但是我们马上发现这是具有后效性的。。也就是一个从(1,1)开始走,一个从(n,m)开始走的话

这样在相同的时间点我们就没法判断两个路径是否是相交的
于是在dp写挂了之后。。我们妥妥写了一发爆搜。。vis的那种
一旦你用了vis数组之后。。我们就不能再记忆化搜索了。。因为你缺少记录vis数组的状态。。
去了记忆化。。来了发纯爆搜。。果然T了。。但是在codevs上还得了30分。。不错不错。。
不错个锤子,ACM就是TLE
贴TLE代码
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring> using namespace std;
int n,m;
int a[55][55];
int dp[55][55][55][55];
bool vis[55][55];
const int INF=(~0u)>>2;
bool check(int x,int y){
if(x>=1&&x<=n&&y>=1&&y<=m&&!vis[x][y]) return true;
return false;
}
int dfs(int x1,int y1,int x2,int y2,int dep){
// printf("(%d,%d) (%d,%d) DEP:%d\n",x1,y1,x2,y2,dep);
// if(dp[x1][y1][x2][y2]!=-1) if(vis[x1][y1]||vis[x2][y2]) return -INF;else return dp[x1][y1][x2][y2];
if(x1==x2&&y1==y2) return -INF;
if(x1==n&&y1==m&&x2==1&&y2==1) return 0;
int p=-INF;
if(check(x1+1,y1)&&check(x2-1,y2)){vis[x1+1][y1]=vis[x2-1][y2]=true;p=max(p,dfs(x1+1,y1,x2-1,y2,dep+1));vis[x1+1][y1]=vis[x2-1][y2]=false;}
if(check(x1+1,y1)&&check(x2,y2-1)){vis[x1+1][y1]=vis[x2][y2-1]=true;p=max(p,dfs(x1+1,y1,x2,y2-1,dep+1));vis[x1+1][y1]=vis[x2][y2-1]=false;}
if(check(x1,y1+1)&&check(x2-1,y2)){vis[x1][y1+1]=vis[x2-1][y2]=true;p=max(p,dfs(x1,y1+1,x2-1,y2,dep+1));vis[x1][y1+1]=vis[x2-1][y2]=false;}
if(check(x1,y1+1)&&check(x2,y2-1)){vis[x1][y1+1]=vis[x2][y2-1]=true;p=max(p,dfs(x1,y1+1,x2,y2-1,dep+1));vis[x1][y1+1]=vis[x2][y2-1]=false;}
// printf("DP (%d,%d,%d,%d):%d\n",x1,y1,x2,y2,p+a[x1][y1]+a[x2][y2]);
// printf("from (%d,%d,%d,%d),(%d,%d,%d,%d),(%d,%d,%d,%d),(%d,%d,%d,%d)\n",x1+1,y1,x2-1,y2,x1+1,y1,x2,y2-1,x1,y1+1,x2-1,y2,x1,y1+1,x2,y2-1);
return dp[x1][y1][x2][y2]=p+a[x1][y1]+a[x2][y2];
}
int main(){
scanf("%d%d",&n,&m);
int i,j;
for(i=1;i<=n;++i)
for(j=1;j<=m;++j) scanf("%d",&a[i][j]);
memset(dp,-1,sizeof(dp));
printf("%d\n",dfs(1,1,n,m,0));
return 0;
}
我们有一个好办法。。那就是其实我们发现从两边搜到对角。。和从一边搜到对角即(1,1)=>(n,m)+(n,m)=>(1,1)==(1,1)=>(n,m)+(1,1)=>(n,m)
并且我们发现路程相同。。则速度也相同。。因为时间一样就妥妥的了。。
这样搜的话。。手画一下图就能发现。。我们在同一时刻就能判断相交了。。因为时间相同速度相同。。如果终点相同。。等于路程一定,速度不变
则时间是相同的。。
并且我们只需枚举x1,x2,step,因为起点固定在(1,1),则后面的点和起点的关系就是,(x1-1)+(y1-1)=step,即x1+y1=step+2
y1=step+2-x1;同理也有x2,y2
所以这样就能省下一维空间。。(这个感觉有点类似于数据库的范式惹,尽量少的元素决定主键。。主键生成的其他不算数。。
然后我们来贴一发代码
其中判定坐标不是(1,1)应该写成!(x==1&&y==1)或者写成(x!=1||y!=1),千万别丢掉括号!
贴上21ms代码
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring> using namespace std;
int n,m;
int a[55][55];
int dp[55][55][55];
const int INF=(~0u)>>2;
bool check(int x,int y){
if(x>=1&&x<=n&&y>=1&&y<=m) return true;
return false;
}
//相当于把两人放在同一起跑线生成到重点的不相交路线,这样判定相交就容易了很多,两头分别搜容易错开。。用过的后面会访问
//但是从同一点出发,因为速度是相同的,如果相交一定是同时在某一点相交(一定),容易想象不同点交错使用共同点的情况
//考虑到速度相同且起点相同。。仅仅枚举x1,x2就能得出各自的纵坐标,即(x-1)+(y-1)=sum,x+y=sum+2,则y=sum+2-x
int dfs(int x1,int x2,int s){
int y1=s+2-x1,y2=s+2-x2;
// printf("(%d,%d) (%d,%d) DEP:%d\n",x1,y1,x2,y2,dep);
if(dp[x1][x2][s]!=-1) return dp[x1][x2][s];
if(!check(x1,y1)||!check(x2,y2)) {
// printf("PASS BORDER\n");
return dp[x1][x2][s]=-INF;
}
if(s==n+m-2){
// printf("ENOUGH STEP\n");
if(x1==x2&&y1==y2&&x1==n&&y1==m) return dp[x1][x2][s]=0;
return dp[x1][x2][s]=-INF;
}
if(x1==x2&&y1==y2&&(x1!=1||y1!=1)) {//判定特殊点搞错
// printf("JUDGE CROSS POINT\n");
if(x1==n&&y1==m) return dp[x1][x2][s]=0;
return dp[x1][x2][s]=-INF;
}
int p=-INF;
p=max(p,dfs(x1,x2,s+1));
p=max(p,dfs(x1+1,x2,s+1));
p=max(p,dfs(x1,x2+1,s+1));
p=max(p,dfs(x1+1,x2+1,s+1));
// printf("DP (%d,%d,%d,%d):%d\n",x1,y1,x2,y2,p+a[x1][y1]+a[x2][y2]);
// printf("from (%d,%d,%d,%d),(%d,%d,%d,%d),(%d,%d,%d,%d),(%d,%d,%d,%d)\n",x1,s+3-x1,x2,s+3-x2,x1+1,s+3-x1-1,x2,s+3-x2,x1,s+3-x1,x2+1,s+3-x2-1,x1+1,s+3-x1-1,x2+1,s+3-x2-1);
return dp[x1][x2][s]=p+a[x1][y1]+a[x2][y2];
}
int main(){
scanf("%d%d",&n,&m);
int i,j;
for(i=1;i<=n;++i)
for(j=1;j<=m;++j) scanf("%d",&a[i][j]);
memset(dp,-1,sizeof(dp));
printf("%d\n",dfs(1,1,0));
return 0;
}
codevs1169传纸条 不相交路径取最大,四维转三维DP的更多相关文章
- CodeVS1169 传纸条 [DP补完计划]
题目传送门 题目描述 Description 小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题.一次素质拓展活动中,班上同学安排做成一个m行n列的矩阵,而小渊和小轩被安排在矩阵对角线的两端, ...
- Codevs1169:传纸条——题解
题目描述 Description 小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题.一次素质拓展活动中,班上同学安排做成一个m行n列的矩阵,而小渊和小轩被安排在矩阵对角线的两端,因此,他们就 ...
- 算法历练之路——传纸条(JAVA)
传纸条 时间限制: 1Sec 内存限制: 128MB 提交: 36 解决: 16 题目描述小渊和小轩是好朋友也是同班同学,他们在一起 总有谈不完的话题.一次素质拓展活动中,班上同学安排做成一个m行n列 ...
- 四维dp,传纸条,方格取数
四维dp例题 四维dp便是维护4个状态的dp方式 拿题来说吧. 1. 洛谷P1004 方格取数 #include<iostream> #include<cstdio> usin ...
- 【暑假集训】HZOI2019 Luogu P1006 传纸条 二三四维解法
写三次丢失两次,我谔谔,以后再不在博客园先保存我就去死 题目内容 洛谷链接 小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题.一次素质拓展活动中,班上同学被安排坐成一个\(m\)行.\(n\ ...
- TYVJ 1011 NOIP 2008&&NOIP 2000 传纸条&&方格取数 Label:多线程dp
做题记录:2016-08-15 15:47:07 背景 NOIP2008复赛提高组第三题 描述 小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题.一次素质拓展活动中,班上同学安排做成一个m行 ...
- P1011 传纸条//dp优化改进状态表示
P1011 传纸条 时间: 1000ms / 空间: 131072KiB / Java类名: Main 背景 NOIP2008复赛提高组第三题 描述 小渊和小轩是好朋友也是同班同学,他们在一起总有谈不 ...
- Wikioi 1169 传纸条
这道题是我人生第一道双线动规题,因此我觉得还是很有必要记录下来. 刚接触到这道题的时候我第一反应是单线的动规,可是下一秒我就觉得这样做可能会有问题,因为从左上角(以下简称A)到右下角(以下简称B)通过 ...
- 蓝桥杯 算法训练 ALGO-36 传纸条
算法训练 传纸条 时间限制:1.0s 内存限制:512.0MB 问题描述 小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题.一次素质拓展活动中,班上同学安排做成一个m行n列的矩阵,而 ...
随机推荐
- jackson学习之二:jackson-core
欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...
- MySQL索引的原理,B+树、聚集索引和二级索引
MySQL索引的原理,B+树.聚集索引和二级索引的结构分析 一.索引类型 1.1 B树 1.2 B+树 1.3 哈希索引 1.4 聚集索引(clusterd index) 1.5 二级索引(secon ...
- JavaWeb——过滤器及监听器
什么是过滤器? 过滤器示意图 Filter是如何实现拦截的? Filter的生命周期 Filter的创建 Filter的销毁 FilterConfig接口 Servlet过滤器有关接口 过滤器配置 F ...
- Spark SQL 自定义函数类型
Spark SQL 自定义函数类型 一.spark读取数据 二.自定义函数结构 三.附上长长的各种pom 一.spark读取数据 前段时间一直在研究GeoMesa下的Spark JTS,Spark J ...
- 通过f5的默认路由使服务器上网
1.通过f5的默认路由使服务器上网 1)将服务器的默认网关指到f5的floating ip 2)f5上配置
- Java编程工具IDEA的使用
IDEA psvm + Enter 快速构建main方法 sout + Enter 快速打印与句 Ctrl+Shift + Enter,语句完成 Ctrl+F12,可以显示当前文件的结构 Ctrl + ...
- dedecms不能保存jpeg格式图片的解决方法
方法如下: 进入织梦的后台管理目录,默认是dede文件夹,找到/inc/inc_archives_functions.php文件. 在文件里查找gif|jpg|,我找到了4个, 在它们后面加jpeg的 ...
- Flink-v1.12官方网站翻译-P020-Builtin Watermark Generators
内置水印生成器 正如在Generating Watermarks一文中所描述的,Flink提供了抽象,允许程序员分配自己的时间戳和发射自己的水印.更具体地说,可以通过实现WatermarkGenera ...
- Jenkins(8)构建触发器之定时构建和轮询 SCM
前言 跑自动化用例每次用手工点击jenkins出发自动化用例太麻烦了,我们希望能每天固定时间跑,这样就不用管了,坐等收测试报告结果就行. jenkins的定时任务是用的crontab语法 定时构建语法 ...
- PTA甲级—链表
1032 Sharing (25分) 回顾了下链表的基本使用,这题就是判断两个链表是否有交叉点. 我最开始的做法就是用cnt[]记录每个节点的入度,发现入度为2的节点即为答案.后来发现这里忽略了两个链 ...