题目描述

在一个 n*m 的棋盘上,每个格子有一个权值,初始时,在某个格子的顶点处一只面朝北的蚂蚁,我们只知道它的行走路线是如何转弯,却不知道每次转弯前走了多长。

蚂蚁转弯是有一定特点的,即它的转弯序列一定是如下的形式:右转,右转,左转,左转,右转,右转…左转,左转,右转,右转,右转。即两次右转和两次左转交替出现的形式,最后两次右转(最后两次一定是右转)后再多加一次右转。我们还知道,蚂蚁不会在同一个位置连续旋转两次,并且蚂蚁行走的路径除了起点以外,不会到达同一个点多次,它最后一定是回到起点然后结束自己的行程,而且蚂蚁只会在棋盘格子的顶点处转弯。

现在已知棋盘大小、每个格子的权值以及左转次数/2 的值,问蚂蚁走出的路径围出的封闭图形,权值之和最大可能是多少。

输入输出格式

输入格式:

在输入文件ant.in 中,第一行三个数n,m,k。意义如题目描述。

接下来一个n 行m 列的整数矩阵,表示棋盘。

输出格式:

一个数,表示蚂蚁所走路径围出的图形可能的最大权值和。

输入输出样例

输入样例#1: 复制

2 5 2
-1 -1 -1 -1 -1
-1 -1 -1 -1 -1
输出样例#1: 复制

-8

说明

【样例说明】

除了第一行的第二个和第一行的第四个都要围起来才至少合法。

【数据规模与约定】

10%的数据所有格子中权值均非负

另20%的数据n=2

另30%的数据k=0

100%的数据1≤n≤100,1≤m≤100,0≤k≤10 保证存在合法路径,数据有梯度,格子中每个元素的值绝对值不超过 10000

P3335 这个题思维难度还是有的。。(至少我是这么想的。。大佬就别吐槽我了)

首先,通过题目描述,我们可以在纸上画一画,可以发现,图像一定是像长城一样的

就是好多个矩形它们的底在一条直线上,高和宽不同,而且,还有一点就是它是高低相间的

而且由右转形成高峰,由左转形成低谷。

那么我们可以枚举图的右下角(i,j),那么有两种情况:

一:第j-1列和第j列在同一个矩形里;

二:第j-1列和第j列在不同的矩形里;

我们要记录的状态与点(i,j),p(指的是当前枚举的是第p个矩形),h(当前枚举的举行高度为i-h+1)有关

所以用f[i][j][p][h]来记录‘一’情况的状态,用g[i][j][p][h][0/1]来记录‘二’情况的状态

这里0表示上一个矩形高度高于h,1表示低于h;

那么转移就好写了:

f[i][j][p][h]=max(f[i][j-1][p][h],g[i][j-1][p-1][h][p%2])+s[i][j]-s[h-1][j];

对了,这里这个s数组求的是每一列的前缀和,可以在输入中预处理出来,方便计算用;

关于g数组的维护,我们已经维护出f数组的第j列了

那么这一列所在的矩形要么是低谷,要么是高峰,我们都要考虑

->高峰:

g[i][j][p][h][0]=max(f[i][j][p][h-1],g[i][j][p][h-1][0]);

->低谷:

g[i][j][p][h][1]=max(f[i][j][p][h+1],g[i][j][p][h+1][1]);

当然我们可以在计算过程中更新答案,还可以省掉i这一维

因为从方程中就可以看出来i其实没有参与转移

 #pragma GCC optimize(2)
#pragma G++ optimize(2)
#pragma GCC target ("avx")
#include<iostream>
#include<algorithm>
#include<cmath>
#include<cstdio>
#include<cstring>
using namespace std;
const int maxn=;
const int Inf=;
int n,m,k,ans;
int a[maxn][maxn];
int f[maxn][][maxn];
int g[maxn][][maxn][];
int s[maxn][maxn];
void ini()
{
scanf("%d%d%d",&n,&m,&k);
//因为有2*k次左转,所以总矩形数就是k*2+1
k=k*+;
for(int i=;i<=n;i++)
{
for(int j=;j<=m;j++)
{
scanf("%d",&a[i][j]);
s[i][j]=s[i-][j]+a[i][j];
}
}
//预处理 因为要求最大值,所以边界赋值为-Inf;
for(int p=;p<=k;p++)
{
for(int h=;h<=n;h++)
{
f[][p][h]=-Inf;
g[][p][h][]=-Inf;
g[][p][h][]=-Inf;
}
}
}
void dp()
{
ans=-Inf;
for(int i=;i<=n;i++)
{
for(int j=;j<=m;j++)
{
for(int p=;p<=k;p++)
{
for(int h=i;h>=;h--)//维护f数组
{
f[j][p][h]=max(f[j-][p][h],g[j-][p-][h][p%])+s[i][j]-s[h-][j];
}
//维护g数组
g[j][p][][]=-Inf;
//0指当前矩形比下一个高,所以从高到低更新,才可以确保取最大值的矩形一定是高的
for(int h=;h<=i;h++)
{
g[j][p][h][]=max(f[j][p][h-],g[j][p][h-][]);
}
g[j][p][i][]=-Inf;
//1指当前矩形比下一个底,所以从低到高更新,才可以确保取最大值的矩形一定是低的
for(int h=i-;h>=;h--)
{
g[j][p][h][]=max(f[j][p][h+],g[j][p][h+][]);
}
}
//更新答案,因为最后一列一定是高的,所以用0转移;
ans=max(ans,max(f[j][k][i],g[j][k][i][]));
}
}
}
int main()
{
ini();//读入一些数据
dp();
printf("%d\n",ans);
return ;
}

bzoj 3111 蚂蚁 动态规划的更多相关文章

  1. BZOJ 3111: [Zjoi2013]蚂蚁寻路

    Sol DP. 首先观察转折,画画图,看看移动路线,可以非常轻易的发现如果走到起点的下方是回不去的.. 然后它就相当于一个底部是平的,顶部凹凹凸凸的形状,每右转两次或左转两次就会形成小矩阵,这样就可以 ...

  2. bzoj 5185 Lifeguards - 动态规划 - 贪心

    题目传送门 传送点I 传送点II 题目大意 给定$n$个区间,问恰好删去其中$k$个,剩下的区间的并的最大总长度. 显然被包含的区间一定不优.再加上被包含的区间对计数不友好.直接把它删掉. 注意到题目 ...

  3. bzoj 4621: Tc605 动态规划

    题解: 一道比较简单的题目 想着想着就把题目记错了..想成了可以把某段区间覆盖为其中一个数 其实是比较简单的 每个点的贡献一定是一个区间(就跟zjoi2018那题一样) 然后问题就变成了给你n个区间让 ...

  4. bzoj 4318 OSU! - 动态规划 - 概率与期望

    Description osu 是一款群众喜闻乐见的休闲软件.  我们可以把osu的规则简化与改编成以下的样子:  一共有n次操作,每次操作只有成功与失败之分,成功对应1,失败对应0,n次操作对应为1 ...

  5. 3111: [Zjoi2013]蚂蚁寻路 - BZOJ

    题目描述 Description在一个 n*m 的棋盘上,每个格子有一个权值,初始时,在某个格子的顶点处一只面朝北的蚂蚁,我们只知道它的行走路线是如何转弯,却不知道每次转弯前走了多长.蚂蚁转弯是有一定 ...

  6. bzoj 4767 两双手 - 动态规划 - 容斥原理

    题目传送门 传送门I 传送门II 题目大意 一个无限大的棋盘上有一只马,设马在某个时刻的位置为$(x, y)$, 每次移动可以将马移动到$(x + A_x, y + A_y)$或者$(x + B_x, ...

  7. bzoj 3522 / 4543 [POI 2014] Hotel - 动态规划 - 长链剖分

    题目传送门 bzoj 3522 需要root权限的传送点 bzoj 4543 快速的传送点 慢速的传送点 题目大意 给定一棵树,问有多少个无序三元组$(x, y, z)$使得这三个不同点在树上两两距离 ...

  8. bzoj 1304 [CQOI 2009] 叶子的染色 - 动态规划

    题目传送门 快速的传送门 慢速的传送门 题目大意 给定一棵无根树,每个点可以染成黑色或者白色,第$i$叶节点到根的路径上最后有颜色的点必须为$c_{i}$(叶节点可以染色).问最少要染颜色的点的个数. ...

  9. bzoj 4044 Virus synthesis - 回文自动机 - 动态规划

    题目传送门 需要高级权限的传送门 题目大意 要求用两种操作拼出一个长度为$n$的只包含'A','T','G','C'的字符串 在当前字符串头或字符串结尾添加一个字符 将当前字符串复制,将复制的串翻转, ...

随机推荐

  1. Salt-ssh 自动安装salt-minion

    作用:为了不手动去安装一台一台去salt-minion,并进重复的配置 一.环境 系统环境: #cat /etc/redhat-release CentOS Linux release 7.4.170 ...

  2. C++指针之间的赋值与转换规则总结

    C++指针之间的赋值与转换规则总结 Note:以下结论不适用于类的成员函数指针,关于类的成员函数指针会单独讨论. 一.任何类型的指针变量均可直接赋值给const void * 任何类型的非const指 ...

  3. 【php】关于trim,rtrim,ltrim,substr 的字符串切割导致 json,_encode无法 识别数据的问题

    示例 <?php $a = rtrim('南宁 .',' .'); echo $a; //输出 南�� echo json_encode($a); //输出空白 $b = ['name'=> ...

  4. java 二进制、位运算、和移位运算符(2013-07-30-bd 写的日志迁移

    二进制是逢2进位的进位制,0.1是基本算符, 1字节=8位 比如 int a =1 ;int 占4个字节在计算机里表示为: java中的4个位运算,分别是“按位与&.按位或|.按位异或^,按位 ...

  5. python中的字典内置方法小结

    #!/usr/local/bin/python3 # -*- coding:utf-8 -*- #key-value #dict 无序,无下标,不需要下标,因为有key stu={ 'stu001': ...

  6. Effective Approaches to Attention-based Neural Machine Translation(Global和Local attention)

    这篇论文主要是提出了Global attention 和 Local attention 这个论文有一个译文,不过我没细看 Effective Approaches to Attention-base ...

  7. PHP.22-Smart模版

    Smart模版 smarty是一个基于PHP开发的PHP模板引擎.它提供了逻辑与外在内容的分离,简单的讲,目的就是要使PHP程序员同美工分离,使用的程序员改变程序的逻辑内容不会影响到美工的页面设计,美 ...

  8. Android开发——View滑动的三种实现方式

    0. 前言   Android开发中,我们常常需要View滑动实现一些绚丽的效果来优化用户体验.一般View的滑动可以用三种方式实现. 转载请注明出处:http://blog.csdn.net/seu ...

  9. 9path 导致的一场冤假错案

    今天做对话框开发,遇到一个问题,就是弹出来的对话框太丑了.如图: 大家都是warp_content, 前面几个就是真的wrap_coment了.只有最后一个还可以看.后来自己找代码,写的都一样,就去问 ...

  10. CentOS 7 安装Oracle 11gR2

    概述 Oracle 在Linux和window上的安装不太一样,公司又是Linux系统上的Oracle,实在没辙,研究下Linux下Oracle的使用,oracle默认不支持CentOS系统安装,所以 ...