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

题目内容

洛谷链接

小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题。一次素质拓展活动中,班上同学被安排坐成一个\(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\)。或者手动判断重复的时候减去一个。

代码

#include <bits/stdc++.h>
using namespace std;
const int maxn=55;
int n,m;
int f[maxn][maxn],a[maxn][maxn]; int mymax(int a,int b,int c,int d){
return max(max(max(a,b),c),d);
} int main(){
scanf("%d%d",&n,&m); for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
scanf("%d",&a[i][j]); for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
for(int k=1;k<=n;k++)
for(int q=j+1;q<=m;l++)
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]; printf("%d",f[n][m-1][n-1][m]);
return 0;
}

式子太长辣有可能鬼畜辣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的美滋滋,同时要手动判重。

代码

#include <bits/stdc++.h>
using namespace std;
const int maxn=55;
int n,m;
int f[2*maxn][maxn][maxn],a[maxn][maxn];
int mymax(int a,int b,int c,int d){
return max(max(max(a,b),c),d);
}
int main(){
scanf("%d%d",&n,&m);
for (int i=1;i<=n;i++)
for (int j=1;j<=m;j++)
scanf("%d",&a[i][j]); for (int k=1;k<=n+m-1;k++)
for (int i=1;i<=n;i++)
for (int j=1;j<=n;j++){
if (k-i+1<1||k-j+1<1)continue;//判断纵坐标的合法性
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];
if(i==j)//重合删掉一个(若数据中有负数则不能这么判重!)
f[k][i][j]-=a[i][k-i+1];
}
printf("%d\n",f[n+m-1][n][n]);
return 0;
}

二维

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

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

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

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

代码

#include <bits/stdc++.h>
using namespace std;
const int maxn=210;
int n,m;
int f[maxn][maxn],a[maxn][maxn]; int mymax(int a,int b,int c,int d){
return max(max(max(a,b),c),d);
} int main(){
//freopen("1.txt","r",stdin);
scanf("%d%d",&n,&m);
for (int i=1;i<=n;i++)
for (int j=1;j<=m;j++)
scanf("%d",&a[i][j]); for (int k=1;k<=n+m-1;k++)
for (int i=n;i>=1;i--)
for (int j=n;j>i;j--)
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]; printf("%d\n",f[n-1][n]);
return 0;
}

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

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

#include <bits/stdc++.h>
using namespace std;
const int maxn=55;
int m,n;
int a[maxn][maxn];
int f[maxn][maxn][maxn][maxn]; int main(){
scanf("%d%d",&m,&n);
for(int i=1;i<=m;i++)
for(int j=1;j<=n;j++)
scanf("%d",&a[i][j]); for(int i=1;i<=m;i++){
for(int j=1;j<=n;j++){
for(int k=1;k<=m&&i+j>k;k++){
int q=i+j-k;
if(i==k)
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];
else
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];
}
}
} printf("%d",f[m][n][m][n]);
return 0;
}

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

【暑假集训】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. Vue 下拉刷新及无限加载组件

    原文  https://github.com/wangdahoo/vue-scroller 主题 Vue.js Vue Scroller Vue Scroller is a foundational ...

  2. windows-android-appium环境搭建

    一.安装jdk 安装jdk1.7以上版本,会生成一个jdk目录,和单独的jre目录(注意:不是jdk里面的jre,时安装过程中设置的那个jre路径)安装完成后并配置环境变量 在系统环境变量中,新建:J ...

  3. ES6语法笔记

    迭代器 // log let arr = ['一', '二', '三'] let iter = arr[Symbol.iterator]() console.log(iter.next()) cons ...

  4. JS基础回顾_Dom

    方法 reverse 将子元素位置颠倒 <!-- log --> <ul id="reverse"> <li>1</li> < ...

  5. js图形打印

    1. 打印等边三角形 document.writeln("打印三角形</br>"); for(var i=0;i<5;i++){ for(var j=5;j> ...

  6. 阿里云docker部署mysql

    看完我的上一个博客之后,对centos系统应该有一定的了解,话不多说,接下来我们来在docker容器中部署mysql. 1.下载mysql镜像,因为本人用的5.7版本,你也可以下载最新版,都是可以的 ...

  7. Shell学习(三)Shell参数传递

    一.传参实例 ##脚本文件内容 #执行的文件名 echo $0; #第一个参数 echo $1; #第二个参数 echo $2; #第三个参数 echo $3; ##调用语句 ./testShell. ...

  8. kafka学习(四)kafka安装与命令行调用

    文章更新时间:2020/06/07 一.安装JDK 过程就不过多介绍了... 二.安装Zookeeper 安装过程可以参考此处~ 三.安装并配置kafka Kafka下载地址  http://kafk ...

  9. 使用SpringBoot的方式配置过滤器

    springboot 不存在web.xml 那么如何配置过滤器呢 springboot提供了一种方式 通过spring容器配置 @Bean public FilterRegistrationBean ...

  10. 《Mybatis进阶》肝了30天专栏文章,整理成册,免费获取!!!

    持续原创输出,点击上方蓝字关注我吧 目录 前言 简介 如何获取? 总结 前言 Mybatis专栏文章写到至今已经有一个月了,从基础到源码详细的介绍了每个知识点,没什么多余的废话,全是工作.面试中常用到 ...