传纸条详解:

蒟蒻最近接到了练习DP的通知,于是跑来试炼场看看;发现有点难(毕竟是蒟蒻吗)便去翻了翻题解,可怎么都看不懂。为什么呢?蒟蒻发现题解里都非常详细的讲了转移方程,讲了降维优化,但这题新颖之处在于它走了两次,可大家貌似都没有重点去讲如何去重啊!

虽然去重很简易,限制一个for循环的范围就行了,但如果没注意这一点,很难理解。这里题解几乎都是for循环里写了几个k>j, j=i+1...然后都不注释一下就开始状态转移了。

所以,本题解诞生了:

写在前面:

P1004 方格取数

如果你觉得此题有些难可以先去看看这道题,他的题面相对更简洁易懂,数据范围也非常小,可以去练练与本题相同的四维的解法。双倍经验啊!

基础:

四维DP,复杂度O(n^4)左右(空间也一样)

用f[i][j][p][q]表示第一张纸条传到(i,j),第二张纸条传到(p,q)所累计下来的好心程度和。转移方程其他题解已经很详细了吧(还是码一下吧...):

对于每一步有四种情况:

1.第一张纸条向下传,第二张纸条向下传;

2.第一张纸条向下传,第二张纸条向右传;

3.第一张纸条向右传,第二张纸条向下传;

4.第一张纸条向右传,第二张纸条向右传;

f[i][j]=max(f[i-1][j][p-1][q] ,f[i-1][j][p][q-1] ,f[i][j-1][p-1][q] ,f[i-1][j][p][q-1])+v[i][j]+v[p][q];

那么如何判重呢?这里其实可以不判,只要你没有重复情况就行了,所以for循环时我们限制p>q即可。

提高:

三维DP,复杂度O(n^3)(空间会多一倍)

我们发现每一张纸条每一步要么只走右边,要么只走下边,所以i+j=p+q;于是我们DP每一步(用k表示)的情况 ,用i表示第一张纸往下走了多少步,因为枚举了k=i+j(即走了多少步)所以可以用k-i来代替j。第二张纸也同样可以用k和p表示出来坐标。因为枚举的是步数(n+m-2)所以空间会多一倍。

于是 F[k][i][p]=max{F[k-1][i][p]+F[k-1][i][p-1]+F[k-1][i-1][p]+F[k-1][i-1][p-1];

进阶:

二维DP,复杂度和三维一样,但空间少了很多

如果你对背包掌握得足够优秀(不像我那么菜),你就能用背包思想来降维。怎么做到的呢?

我们从三维DP的状态转移式中发现它只和上一步有关,还只牵扯到P,P-1,没用到P+1.所以我们从后向前推,这样你现在用的二维数组就是上一步的,对P进行覆盖也不会产生后效性。

那重点来了这又如何去重呢?其实你只需要保证 p > i 就行了,因为这样就不会有重复情况出现,自然也不需要去重了。

代码如下:

#include<iostream>
#include<cstdio>
#include<iomanip>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<cstdlib>
#include<queue>
#include<stack>
#include<vector>
#include<map>
#include<set> #define ll long long
#define db double
#define inf 0x7fffffff
#define init inline int using namespace std; int f[201][201];
int v[201][201];
int n,m; init qr(){
char ch;
while((ch=getchar())<'0'||ch>'9');
int res=ch^48;
while((ch=getchar())>='0'&&ch<='9')
res=res*10+(ch^48);
return res;
} init max(int a,int b,int c,int d){
a=a>b?a:b;
c=c>d?c:d;
return a>c?a:c;
} int main(){
//freopen(".in","r",stdin);
//freopen(".out","w",stdout);
n=qr(),m=qr();
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
v[i][j]=qr();
for(int k=3;k<=n+m;k++)
for(int i=n;i>=1;i--)
for(int p=n;p>i;p--)
f[i][p]=max(f[i][p],f[i-1][p-1],f[i-1][p],f[i][p-1]),
f[i][p]+=v[i][k-i]+v[p][k-p];
printf("%d\n",f[n-1][n]);
return 0;
}

不太想极端压行了(码字累了),代码风格就这样了,不喜勿喷,谢谢了。

然后解释一下输出 f[n-1][n] 是因为j>i的去重需要。

洛谷 P1006 传纸条 多维DP的更多相关文章

  1. 洛谷P1006 传纸条 (棋盘dp)

    好气,在洛谷上交就过了,在caioj上交就只有40分 之前在51nod做过这道题了. https://blog.csdn.net/qq_34416123/article/details/8180902 ...

  2. 棋盘DP三连——洛谷 P1004 方格取数 &&洛谷 P1006 传纸条 &&Codevs 2853 方格游戏

    P1004 方格取数 题目描述 设有N $\times N$N×N的方格图(N $\le 9$)(N≤9),我们将其中的某些方格中填入正整数,而其他的方格中则放入数字00.如下图所示(见样例): A ...

  3. 洛谷 P1006 传纸条 题解

    P1006 传纸条 题目描述 小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题.一次素质拓展活动中,班上同学安排做成一个m行n列的矩阵,而小渊和小轩被安排在矩阵对角线的两端,因此,他们就无法 ...

  4. 洛谷P1006 传纸条(多维DP)

    小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题.一次素质拓展活动中,班上同学安排做成一个mm行nn列的矩阵,而小渊和小轩被安排在矩阵对角线的两端,因此,他们就无法直接交谈了.幸运的是,他们 ...

  5. [NOIP2008] 提高组 洛谷P1006 传纸条

    题目描述 小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题.一次素质拓展活动中,班上同学安排做成一个m行n列的矩阵,而小渊和小轩被安排在矩阵对角线的两端,因此,他们就无法直接交谈了.幸运的是 ...

  6. P1006 传纸条 多维DP

    题目描述 小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题.一次素质拓展活动中,班上同学安排做成一个mm行nn列的矩阵,而小渊和小轩被安排在矩阵对角线的两端,因此,他们就无法直接交谈了.幸运 ...

  7. 洛谷 P1006 传纸条

    题目描述 小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题.一次素质拓展活动中,班上同学安排做成一个m行n列的矩阵,而小渊和小轩被安排在矩阵对角线的两端,因此,他们就无法直接交谈了.幸运的是 ...

  8. 洛谷P1006传纸条

    题目描述 小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题.一次素质拓展活动中,班上同学安排做成一个 m 行 n 列的矩阵,而小渊和小轩被安排在矩阵对角线的两端,因此,他们就无法直接交谈了. ...

  9. 洛谷p1006 传纸条 三维解法

    原题目如下 原地址https://www.luogu.com.cn/problem/P1006 题目描述 小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题.一次素质拓展活动中,班上同学安排做 ...

随机推荐

  1. Async 异步转同步详细流程解释

      安装 npm install async --save 地址 https://github.com/caolan/async Async的内容主要分为三部分 流程控制: 简化九种常见的流程的处理 ...

  2. Notes of Daily Scrum Meeting(12.21)

    今天的燃尽图把周六的进度加了进来,由于我的失误没有及时更新TFS,所以出现了一些错误,向大家道歉. 下面是今天的任务总结: 团队成员 今日团队工作 陈少杰 继续进行网络连接的调试 王迪 测试搜索的功能 ...

  3. idea 使用 mybaits generator

    Intellij IDEA 14 作为JavaIDE 神器,接触后发现,非常好用,对它爱不释手,打算离开eclipse和myeclipse,投入Intellij IDEA的怀抱. 然而在使用的过程中会 ...

  4. Java运算符、switch、数组、排序

    1.Java的运算符,分为四类:算数运算符.关系运算符.逻辑运算符.位运算符 运算符例子:22.25(十进制转化为二进制,8421码)0010 0010 (22)0010 0101 (25) 位运算符 ...

  5. 『编程题全队』Alpha 阶段冲刺博客Day6

    1.每日站立式会议 1.会议照片 2.昨天已完成的工作统计 孙志威: 1.添加JSON处理模块 2.添加了团队看板中的添加团队任务功能 3.添加了团队看板中的添加按钮 孙慧君: 1.个人任务框UI的设 ...

  6. CentOS7 安装redis 并且设置成服务自动启动

    通过 博客园 https://www.cnblogs.com/zuidongfeng/p/8032505.html 学习以及记录 1. 下载redis 现在最新的stable版本是 4.0.10 wg ...

  7. 一本通1640C Looooops

    1640:C Looooops 时间限制: 1000 ms         内存限制: 524288 KB [题目描述] 原题来自:CTU Open 2004 对于 C 语言的 for (variab ...

  8. Grafana elasticsearch 应用

    早期的时候,项目基于ES+echart写了一些仪表盘的展示页面,虽然ES配合这种char界面有着天然的优势,但实际写起代码来,还是很多重复的劳动,在一次偶然中发现Grafana,看到它提供了很多仪表盘 ...

  9. Jenkins+Jmeter+Ant自动化集成环境搭建

    1.搭建环境,安装以下工具 JDK:jdk1.7.0_79 Ant:apache-ant-1.9.7 Jmeter: apache-jmeter-3.0 Jenkins: jenkins-1.651. ...

  10. [代码]--SQLServer数据库的一些全局变量

    select APP_NAME ( ) as w --当前会话的应用程序 select @@IDENTITY   --返回最后插入的标识值 select USER_NAME()    --返回用户数据 ...