【暑假集训】HZOI2019 Luogu P1006 传纸条 二三四维解法
写三次丢失两次,我谔谔,以后再不在博客园先保存我就去死
题目内容
洛谷链接
小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题。一次素质拓展活动中,班上同学被安排坐成一个\(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 传纸条 二三四维解法的更多相关文章
- [Luogu P1006]传纸条 (网格DP)
题面 传送门:https://www.luogu.org/problemnew/show/P1006 Solution 挺显然但需要一定理解的网络(应该是那么叫吧)DP 首先有一个显然但重要的结论要发 ...
- LuoGu P1006 传纸条
题目传送门 这题嘛...方格取数和这题一样一样的 只不过这题是从左上到右下再回去罢了(来回一趟和来两趟有区别么?没有,那么这题和上题用一样的转移和状态就行了 没什么好说的,说一下我的错误好了: 人家图 ...
- Luogu 1006 传纸条 / NOIP 2008 传纸条(动态规划)
Luogu 1006 传纸条 / NOIP 2008 传纸条(动态规划) Description 小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题.一次素质拓展活动中,班上同学安排做成一个m ...
- 【洛谷】【动态规划(多维)】P1006 传纸条
[题目描述:] 小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题.一次素质拓展活动中,班上同学安排做成一个m行n列的矩阵,而小渊和小轩被安排在矩阵对角线的两端,因此,他们就无法直接交谈了.幸 ...
- 棋盘DP三连——洛谷 P1004 方格取数 &&洛谷 P1006 传纸条 &&Codevs 2853 方格游戏
P1004 方格取数 题目描述 设有N $\times N$N×N的方格图(N $\le 9$)(N≤9),我们将其中的某些方格中填入正整数,而其他的方格中则放入数字00.如下图所示(见样例): A ...
- 洛谷 P1006 传纸条 题解
P1006 传纸条 题目描述 小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题.一次素质拓展活动中,班上同学安排做成一个m行n列的矩阵,而小渊和小轩被安排在矩阵对角线的两端,因此,他们就无法 ...
- P1006 传纸条(二维、三维dp)
P1006 传纸条 输入输出样例 输入 #1 复制 3 3 0 3 9 2 8 5 5 7 0 输出 #1 复制 34 说明/提示 [限制] 对于 30% 的数据,1≤m,n≤10: 对于 100% ...
- 洛谷p1006 传纸条 三维解法
原题目如下 原地址https://www.luogu.com.cn/problem/P1006 题目描述 小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题.一次素质拓展活动中,班上同学安排做 ...
- [NOIP2008] 提高组 洛谷P1006 传纸条
题目描述 小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题.一次素质拓展活动中,班上同学安排做成一个m行n列的矩阵,而小渊和小轩被安排在矩阵对角线的两端,因此,他们就无法直接交谈了.幸运的是 ...
随机推荐
- SpringIOC初始化过程--详解
SpringIOC初始化过程 相信大家都知道Spring这个东西,我们经常来用他一些特性,比如说他的AOP,IOC,那今天就带大家解析下SpringIOC的加载过程. 我们来看一个例子 Annotat ...
- python3 while循环
python不支持n++这样格式,因为python中变量不像c那样事先定义好变量类型,在内存中开辟指定的空间,然后赋值. python中以字符串为例,事先在内存划分空间来存放字符串,然后用变量名来指向 ...
- Fiddler的安装和APP抓包
前言 1.Fiddler安装包 2.安卓手机 3.iOS手机 1.下载fiddler软件:可以去官网下载https://www.telerik.com/fiddler,可以下载最新版 2.百度云盘(非 ...
- 利用 QEMU USER 模式运行 mips 程序
摘要 关键字: qemu mips 前述 QEMU是一个处理器模拟软件,可以用来在PC中模拟ARM.MIPS等多种架构的软硬件运行环境.QEMU主要有两种模拟模式: User Mode System模 ...
- JVM运行时数据区--方法区
运行时数据区结构图(温习): 堆.栈.方法区的交互关系 方法区的理解 方法区(Method Area)与Java堆一样,是各个线程共享的内存区域 方法区在JVM启动时就会被创建,并且它的实际的物理内存 ...
- Java一键部署包,Linux部署不用愁!!!
前言 昨天一哥们的弟弟突然问我有没有部署过的Linux,公司连个运维都没有,服务器都要后端部署.... 你有没有相似的遭遇呢?公司规模小,后端即是运维,一份工资干两份活,哈哈~ 为了解决这老弟的困惑, ...
- Linux实战(6):Centos8上传镜像
小记 做以下操作之前得准备一个Docker Hub 的账号,然后创建仓库可私有也可公有这得看你自己了,仓库的名称需记住等等会有用. 安装podman podman跟docker使用的命令非常的接近,但 ...
- 前端直传文件到aliyun OSS
<template> <div id="container"> <div class="img-item m-1 upload-file&q ...
- CVE-2020-0796(Windows SMBv3) RCE漏洞复现
CVE-2020-0796 攻击机:win10:192.168.205.1 靶机win10:192.168.205.132 关闭defender防火墙 0x01 影响版本 Windows 10 190 ...
- 【转】Locust性能-零基础入门系列(1)-wait_time属性用法
本篇文章,从局部出发,利用一个简单的测试,来说明场景模拟的wait_time属性的用法.wait_time为什么要单独拎出来讲,是因为它主要有两种模式,而初学者对这两种模式,容易混淆.1) wait_ ...