题目大意

有一个\(n\times m\)的网格,\((x,y)\)权值为\(a_{x,y}\),要求从中选取三个不相交的\(k\times k\)的正方形使得它们权值最大。\(n,m,k\leqslant1500\)

题解

其实,只有如下六种方法分割网格:

对于每一种情况,我们在每个小方格中找最大的\(k\times k\)的正方形相加即可。可以令\(a[i][j],b[i][j],c[i][j],d[i][j]\)分别表示\((i,j)\)的左上、右上、左下、右下的区域中最大的\(k\times k\)的正方形的权值,然后就可以计算了

卡点

C++ Code:

#include <cstdio>
#include <iostream>
#include <algorithm>
#include <cctype>
const int maxn = 1510;
namespace io {
struct istream {
#define M (1 << 24)
char buf[M], *ch = buf - 1;
inline istream() { fread(buf, 1, M, stdin); }
inline istream& operator >> (int &x) {
while (isspace(*++ch));
for (x = *ch & 15; isdigit(*++ch); ) x = x * 10 + (*ch & 15);
return *this;
}
#undef M
} cin;
} int n, m, k, ans, s[maxn][maxn];
int a[maxn][maxn], b[maxn][maxn], c[maxn][maxn], d[maxn][maxn];
/*
* a | b
* - - -
* c | d
*/
int main() {
std::ios::sync_with_stdio(false), std::cin.tie(0), std::cout.tie(0);
io::cin >> n >> m >> k;
for (int i = 1; i <= n; ++i)
for (int j = 1; j <= m; ++j)
io::cin >> s[i][j];
for (int i = 1; i <= n; ++i)
for (int j = 1; j <= m; ++j)
s[i][j] += s[i][j - 1];
for (int i = 1; i <= n; ++i)
for (int j = 1; j <= m; ++j)
s[i][j] += s[i - 1][j];
for (int i = n; i >= k; --i)
for (int j = m; j >= k; --j)
s[i][j] += s[i - k][j - k] - s[i - k][j] - s[i][j - k];
for (int i = k; i <= n; ++i)
for (int j = k; j <= m; ++j)
a[i][j] = std::max({ s[i][j], a[i - 1][j], a[i][j - 1] });
for (int i = k; i <= n; ++i)
for (int j = m - k + 1; j; --j)
b[i][j] = std::max({ s[i][j + k - 1], b[i - 1][j], b[i][j + 1] });
for (int i = n - k + 1; i; --i)
for (int j = k; j <= m; ++j)
c[i][j] = std::max({ s[i + k - 1][j], c[i + 1][j], c[i][j - 1] });
for (int i = n - k + 1; i; --i)
for (int j = m - k + 1; j; --j)
d[i][j] = std::max({ s[i + k - 1][j + k - 1], d[i + 1][j], d[i][j + 1] });
for (int i = k; i <= n - k; ++i)
for (int j = k; j <= m - k; ++j)
ans = std::max(ans, a[i][j] + b[i][j + 1] + c[i + 1][m]);
for (int i = k; i <= n - k; ++i)
for (int j = k; j <= m - k; ++j)
ans = std::max(ans, a[i][m] + c[i + 1][j] + d[i + 1][j + 1]);
for (int i = k; i <= n - k; ++i)
for (int j = k; j <= m - k; ++j)
ans = std::max(ans, a[i][j] + c[i + 1][j] + b[n][j + 1]);
for (int i = k; i <= n - k; ++i)
for (int j = k; j <= m - k; ++j)
ans = std::max(ans, a[n][j] + b[i][j + 1] + d[i + 1][j + 1]);
for (int i = k; i <= n; ++i)
for (int j = 2 * k; j <= m - k; ++j)
ans = std::max(ans, s[i][j] + a[n][j - k] + b[n][j + 1]);
for (int i = 2 * k; i <= n - k; ++i)
for (int j = k; j <= m; ++j)
ans = std::max(ans, s[i][j] + a[i - k][m] + c[i + 1][m]);
std::cout << ans << '\n';
return 0;
}

[SOJ #686]抢救(2019-11-7考试)/[洛谷P3625][APIO2009]采油区域的更多相关文章

  1. 洛谷P3625 - [APIO2009]采油区域

    Portal Description 给出一个\(n\times m(n,m\leq1500)\)的矩阵,从中选出\(3\)个互不相交的\(k\times k\)方阵,使得被选出的数的和最大. Sol ...

  2. 洛谷 P3625 [APIO2009]采油区域【枚举】

    参考:https://blog.csdn.net/FAreStorm/article/details/49200383 没有技术含量但是难想难写,枚举情况图详见参考blog懒得画了 bzoj蜜汁TTT ...

  3. BZOJ1178或洛谷3626 [APIO2009]会议中心

    BZOJ原题链接 洛谷原题链接 第一个问题是经典的最多不相交区间问题,用贪心即可解决. 主要问题是第二个,求最小字典序的方案. 我们可以尝试从\(1\to n\)扫一遍所有区间,按顺序对每一个不会使答 ...

  4. BZOJ1179或洛谷3672 [APIO2009]抢掠计划

    BZOJ原题链接 洛谷原题链接 在一个强连通分量里的\(ATM\)机显然都可被抢,所以先用\(tarjan\)找强连通分量并缩点,在缩点的后的\(DAG\)上跑最长路,然后扫一遍酒吧记录答案即可. # ...

  5. 【题解】洛谷P3627 [APIO2009]抢掠计划(缩点+SPFA)

    洛谷P3627:https://www.luogu.org/problemnew/show/P3627 思路 由于有强连通分量 所以我们可以想到先把整个图缩点 缩点完之后再建一次图 把点权改为边权 并 ...

  6. 洛谷 P3627 [APIO2009]抢掠计划 Tarjan缩点+Spfa求最长路

    题目地址:https://www.luogu.com.cn/problem/P3627 第一次寒假训练的结测题,思路本身不难,但对于我这个码力蒟蒻来说实现难度不小-考试时肛了将近两个半小时才刚肛出来. ...

  7. 洛谷 P3627 [APIO2009]抢掠计划

    这题一看就是缩点,但是缩完点怎么办呢?首先我们把所有的包含酒吧的缩点找出来,打上标记,然后建立一张新图, 每个缩点上的点权就是他所包含的所有点的点权和.但是建图的时候要注意,每一对缩点之间可能有多条边 ...

  8. 洛谷 P3627 [APIO2009](抢掠计划 缩点+spfa)

    题目描述 Siruseri 城中的道路都是单向的.不同的道路由路口连接.按照法律的规定, 在每个路口都设立了一个 Siruseri 银行的 ATM 取款机.令人奇怪的是,Siruseri 的酒吧也都设 ...

  9. [洛谷P3627][APIO2009]抢掠计划

    题目大意:给你一张$n(n\leqslant5\times10^5)$个点$m(m\leqslant5\times10^5)$条边的有向图,有点权,给你起点和一些可能的终点.问从起点开始,到任意一个终 ...

随机推荐

  1. cpio建立、还原备份档

    1. 简介 加入.解开cpio或tar备份档内的文件 与tar相似,将文件归档到硬盘或磁带等存储设备中 2. tar比较 在所处理的文件类型方面,它比tar更全面,但也更复杂 cpio比tar更为可靠 ...

  2. Java进程间通信学习

    转自:https://www.iteye.com/blog/polim-1278435 进程间通信的主要方法有:(1)管道(Pipe):管道可用于具有亲缘关系进程间的通信,允许一个进程和另一个与它有共 ...

  3. FPM十:FORM Repeater

    1.创建feeder class:ZCL_FPM_FORM_REP_DEMO 添加接口类: 2.新建全局内表: DATA:GT_QPGT TYPE TABLE OF QPGT. 3.实列化get_da ...

  4. 第二篇Scrum冲刺博客

    第二篇Scrum冲刺博客 一.站立式会议 提供当天站立式会议照片一张 二.每个人的工作 成员 已完成工作 明天计划完成的工作 遇到的困难 林剑峰 初步学习小程序的编写.博客园的撰写 初步完成用户界面 ...

  5. Java集合目录

    java集合学习(1):集合框架 java集合学习(2):Map和HashMap Java集合学习(4):HashTable Java集合学习(5):LinkedHashMap Java集合学习(6) ...

  6. Linux的rpm管理

                            书山有路勤为径,学海无涯苦作舟 自学linux已经有几天了,感觉还可以.坚持下去,就会有收获. 每个系统都用相应的软件的管理,Linux也不例外.下面讲 ...

  7. Qemu-4.1 桥接网络设置

    参考: [qemu] qemu旧的net参数已经不再可用了,新的这样用. QEMU's new -nic command line option 用Qemu模拟vexpress-a9 --- 配置 q ...

  8. 05、ip划分+网络配置+虚拟化基础+基本路由

    -- IP   IANA (Internet Assigned Numbers Authority) ,Internet号分配机构.负责对IP地 址分配规划以及对TCP/UDP公共服务的端口定义.国际 ...

  9. JavaScript中数组去重汇总

    1. 简单的去重方法,利用数组的indexOf下标属性来查询 /* * 新建一新数组,遍历传入数组,值不在新数组就push进该新数组中 * IE8以下不支持数组的indexOf方法 * */ func ...

  10. PAT 乙级 1032.挖掘机技术哪家强 C++/Java

    题目来源 为了用事实说明挖掘机技术到底哪家强,PAT 组织了一场挖掘机技能大赛.现请你根据比赛结果统计出技术最强的那个学校. 输入格式: 输入在第 1 行给出不超过 1 的正整数 N,即参赛人数.随后 ...