写三次丢失两次,我谔谔,以后再不在博客园先保存我就去死

题目内容

洛谷链接

小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题。一次素质拓展活动中,班上同学被安排坐成一个\(m\)行、\(n\)列的矩阵,而小渊和小 轩被安排坐在矩阵对角线的两端,因此,他们就无法直接交谈了。幸运的是,他们可以通过传纸条来进行交流。纸条要经由许多同学传到对方手里,小渊坐在矩阵的左上角,坐标\((1,1)\),小轩坐在矩阵的右下角,坐标\((m,n)\)。从小渊传给小轩的纸条只可以向下或者向右传递,从小轩传给小渊的纸条只可以向上或者向左传递。

在活动进行中,小渊希望给小轩传一张纸条,同时希望小轩给他回复。班里的每个同学都可以帮他们传递,但只会帮他们一次,也就是说如果此人在小渊递给小轩纸条的时候帮忙,那么在小轩递给小渊的时候就不会再帮忙。反之亦然

还有一件事情需要注意,全班每个同学愿意帮忙的好心程度有高有低(注意:小渊和小轩的好心程度没有定义,输入时用\(0\)表示),可以用一个\(0-100\)的自然数来表示,数越大表示越好心。小渊和小轩希望尽可能找好心程度高的同学来帮忙传纸条,即找到来回两条传递路径,使得这两条路径上同学的好心程度之和最大。现在,请你帮助小渊和小轩找到这样两条路径。

输入格式

第一行有两个用空格隔开的整数\(m\)和\(n\),表示班里有\(m\)行\(n\)列\((1\le m,n\le 50)\)。

接下来的\(m\)行是一个\(m×n\)的矩阵,矩阵中第\(i\)行\(j\)列的整数表示坐在第\(i\)行\(j\)列的学生的好心程度。每行的\(n\)个整数之间用空格隔开。

输出格式

共一行,包含一个整数,表示来回两条路上参与传纸条的同学的好心程度之和的最大值。

样例输入

3 3

0 3 9

2 8 5

5 7 0

样例输出

34

思路

更加简单的一道P1004 方格取数

双倍经验啊

四维

时间复杂度\(O(n^2×m^2)\),其实已经能水过了因为范围挺小的orz。

用\(f[i][j][k][q]\)表示第一张纸条传到\((i,j)\),第二张纸条传到\((k,q)\)所累计下来的好心程度总和。

对于每一步有四种情况:

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

\(f[i][j][k][q]=max\{f[i][j-1][k-1][q],f[i-1][j][k][q-1],f[i][j-1][k][q-1],f[i-1][j][k-1][q]\}+a[i][j]+a[k][q]\)

注意要求两条路线严格不重合,所以为了防止重复q的范围应该是\(j+1\)到\(m\)。或者手动判断重复的时候减去一个。

代码

  1. #include <bits/stdc++.h>
  2. using namespace std;
  3. const int maxn=55;
  4. int n,m;
  5. int f[maxn][maxn],a[maxn][maxn];
  6. int mymax(int a,int b,int c,int d){
  7. return max(max(max(a,b),c),d);
  8. }
  9. int main(){
  10. scanf("%d%d",&n,&m);
  11. for(int i=1;i<=n;i++)
  12. for(int j=1;j<=m;j++)
  13. scanf("%d",&a[i][j]);
  14. for(int i=1;i<=n;i++)
  15. for(int j=1;j<=m;j++)
  16. for(int k=1;k<=n;k++)
  17. for(int q=j+1;q<=m;l++)
  18. f[i][j][k][q]=mymax(f[i][j-1][k-1][q],f[i-1][j][k][q-1],f[i][j-1][k][q-1],f[i-1][j][k-1][q])+a[i][j]+a[k][q];
  19. printf("%d",f[n][m-1][n-1][m]);
  20. return 0;
  21. }

式子太长辣有可能鬼畜辣QAQ

三维

进阶版

时间复杂度\(O(n^2×(n+m))\)

可以发现每次转移两个纸条走过的路程总是相等的。即\(i+j=k+q\)。

设\(i+j=k+q=step\),我们枚举\(step\),同时枚举第一个人和第二个人的横坐标或者纵坐标,另一个就可以算出来了,例如枚举横坐标。

\(f[k][i][j]=max\{f[k-1][i][j],f[k-1][i-1][j-1],f[k-1][i][j-1],f[k-1][i-1][j]\}+a[i][k-i+1]+a[j][k-j+1]\)

不过要注意的是枚举步数的一维要开两倍大小,否则RE的美滋滋,同时要手动判重。

代码

  1. #include <bits/stdc++.h>
  2. using namespace std;
  3. const int maxn=55;
  4. int n,m;
  5. int f[2*maxn][maxn][maxn],a[maxn][maxn];
  6. int mymax(int a,int b,int c,int d){
  7. return max(max(max(a,b),c),d);
  8. }
  9. int main(){
  10. scanf("%d%d",&n,&m);
  11. for (int i=1;i<=n;i++)
  12. for (int j=1;j<=m;j++)
  13. scanf("%d",&a[i][j]);
  14. for (int k=1;k<=n+m-1;k++)
  15. for (int i=1;i<=n;i++)
  16. for (int j=1;j<=n;j++){
  17. if (k-i+1<1||k-j+1<1)continue;//判断纵坐标的合法性
  18. f[k][i][j]=mymax(f[k-1][i][j],f[k-1][i-1][j-1],f[k-1][i][j-1],f[k-1][i-1][j])+a[i][k-i+1]+a[j][k-j+1];
  19. if(i==j)//重合删掉一个(若数据中有负数则不能这么判重!)
  20. f[k][i][j]-=a[i][k-i+1];
  21. }
  22. printf("%d\n",f[n+m-1][n][n]);
  23. return 0;
  24. }

二维

其实就是滚动数组优化辣,时间不变,但是空间会小很多。

在三维中,我们可以看出状态的转移只和上一行有关,所以可以想到滚动数组。

去重方法就是使\(j>i\)(比较显然吧)。

(不知为何\(maxn\)设成55会WA \(n=50,m=50\) 的点,改成210才行,为了调这个感觉都要成浪费学校评测机资源了)

代码

  1. #include <bits/stdc++.h>
  2. using namespace std;
  3. const int maxn=210;
  4. int n,m;
  5. int f[maxn][maxn],a[maxn][maxn];
  6. int mymax(int a,int b,int c,int d){
  7. return max(max(max(a,b),c),d);
  8. }
  9. int main(){
  10. //freopen("1.txt","r",stdin);
  11. scanf("%d%d",&n,&m);
  12. for (int i=1;i<=n;i++)
  13. for (int j=1;j<=m;j++)
  14. scanf("%d",&a[i][j]);
  15. for (int k=1;k<=n+m-1;k++)
  16. for (int i=n;i>=1;i--)
  17. for (int j=n;j>i;j--)
  18. f[i][j]=mymax(f[i][j],f[i-1][j-1],f[i][j-1],f[i-1][j])+a[i][k-i+1]+a[j][k-j+1];
  19. printf("%d\n",f[n-1][n]);
  20. return 0;
  21. }

写完本篇题解心态已经崩了

UPD:附日常解法(大家都这么写的诶orz):

  1. #include <bits/stdc++.h>
  2. using namespace std;
  3. const int maxn=55;
  4. int m,n;
  5. int a[maxn][maxn];
  6. int f[maxn][maxn][maxn][maxn];
  7. int main(){
  8. scanf("%d%d",&m,&n);
  9. for(int i=1;i<=m;i++)
  10. for(int j=1;j<=n;j++)
  11. scanf("%d",&a[i][j]);
  12. for(int i=1;i<=m;i++){
  13. for(int j=1;j<=n;j++){
  14. for(int k=1;k<=m&&i+j>k;k++){
  15. int q=i+j-k;
  16. if(i==k)
  17. f[i][j][k][q]=max(max(max(f[i-1][j][k-1][q],f[i][j-1][k-1][q]),f[i-1][j][k][q-1]),f[i][j-1][k][q-1])+a[i][j];
  18. else
  19. f[i][j][k][q]=max(max(max(f[i-1][j][k-1][q],f[i][j-1][k-1][q]),f[i-1][j][k][q-1]),f[i][j-1][k][q-1])+a[i][j]+a[k][q];
  20. }
  21. }
  22. }
  23. printf("%d",f[m][n][m][n]);
  24. return 0;
  25. }

最后这个代码块的高亮没了?我谔谔

【暑假集训】HZOI2019 Luogu P1006 传纸条 二三四维解法的更多相关文章

  1. [Luogu P1006]传纸条 (网格DP)

    题面 传送门:https://www.luogu.org/problemnew/show/P1006 Solution 挺显然但需要一定理解的网络(应该是那么叫吧)DP 首先有一个显然但重要的结论要发 ...

  2. LuoGu P1006 传纸条

    题目传送门 这题嘛...方格取数和这题一样一样的 只不过这题是从左上到右下再回去罢了(来回一趟和来两趟有区别么?没有,那么这题和上题用一样的转移和状态就行了 没什么好说的,说一下我的错误好了: 人家图 ...

  3. Luogu 1006 传纸条 / NOIP 2008 传纸条(动态规划)

    Luogu 1006 传纸条 / NOIP 2008 传纸条(动态规划) Description 小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题.一次素质拓展活动中,班上同学安排做成一个m ...

  4. 【洛谷】【动态规划(多维)】P1006 传纸条

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

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

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

  6. 洛谷 P1006 传纸条 题解

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

  7. P1006 传纸条(二维、三维dp)

    P1006 传纸条 输入输出样例 输入 #1 复制 3 3 0 3 9 2 8 5 5 7 0 输出 #1 复制 34 说明/提示 [限制] 对于 30% 的数据,1≤m,n≤10: 对于 100% ...

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

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

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

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

随机推荐

  1. 6.Kafka消息流处理

  2. CVE-2020-0796(Windows SMBv3) RCE漏洞复现

    CVE-2020-0796 攻击机:win10:192.168.205.1 靶机win10:192.168.205.132 关闭defender防火墙 0x01 影响版本 Windows 10 190 ...

  3. 预科班D12

    2020.09.22星期二 预科班D12 学习内容: 一.修改文件的两种方式 1.方案一 思路:(1)先以r形式打开源文件    (2)将源文件内容一次性读入内存中,在内存中修改完毕    (3)以w ...

  4. 容器云平台No.10~通过gogs+drone+kubernetes实现CI/CD

    什么是CI/CD 持续集成(Continous Intergration,CI)是一种软件开发实践,即团队开发成员经常集成它们的工作,通常每个成员每天至少集成一次,也就意味着每天可能会发生多次集成.每 ...

  5. Volatile禁止指令重排序(三)

    Volatile禁止指令重排 计算机在执行程序时,为了提高性能,编译器和处理器常常会对指令重排,一般分为以下三种: 源代码 -> 编译器优化的重排 -> 指令并行的重排 -> 内存系 ...

  6. GTA5整合版

    GTA5mod整合版游戏介绍 GTA5mod整合版游戏是一款完美破解的游戏,玩家能够在游戏中享受最爽快的角色动作扮演玩法,在这里你将是一名强大的黑帮分子,在这个都市中,你将体验最真实的黑帮社会玩法.G ...

  7. Python-鸡兔同笼问题

    鸡兔同笼问题 -- 今有雉兔同笼,上有三十五头,下有九十四足,问雉兔各几何? --鸡和兔在一个笼子里,从上面数,有35个头:从下面数,有94只脚.问笼中各有几只鸡和兔 如何逻辑整理? -- 鸡头和兔子 ...

  8. 用JTable 实现日历

    效果图: 主要思想:日历最核心的功能就是能显示某年某月对应的日期和星期几.因此只要实现传入具体的年份和月份,得到一组存放了日期的数组a[ ]即可.其中数组的大小设置成42,要考虑的问题是当月的第一天对 ...

  9. Spring循环依赖的问题

      什么是循环依赖?就是两个Bean相互引用,比如用@Autowire 相互注入.   那么Spring是如何解决这个问题的呢?在Bean还未完全实例化前(类只实例化了一部分),将bean提前暴露出来 ...

  10. Salesforce Javascript(二) 箭头函数

    本篇参考:https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Functions/Arrow_functions 我们在 ...