想了很久的dp,看了一眼题解之后感觉自己被安排了。

发现从一个矩形中选择三个不相交的正方形一共只有六种取法。

那么我们可以处理出四个值:

$f_{i, j}$分别表示以$(i, j)$为右下角,左下角,右上角,左上角的矩阵中选一个$k*k$正方形的最大值。

这样就可以算出前四种情况,后两种情况只要乱搞就可以了。

时间复杂度$O(nm)$。

Code:

#include <cstdio>
#include <cstring>
using namespace std; const int N = ; int n, m, k, ans, a[N][N], sum[N][N], s[N][N];
int f1[N][N], f2[N][N], f3[N][N], f4[N][N], r[N], c[N]; template <typename T>
inline void read(T &X) {
X = ; char ch = ; T op = ;
for(; ch > ''|| ch < ''; ch = getchar())
if(ch == '-') op = -;
for(; ch >= '' && ch <= ''; ch = getchar())
X = (X << ) + (X << ) + ch - ;
X *= op;
} inline void chkMax(int &x, int y) {
if(y > x) x = y;
} inline int max(int x, int y) {
return x > y ? x : y;
} inline int max(int x, int y, int z) {
return max(max(x, y), z);
} int main() {
read(n), read(m), read(k);
for(int i = ; i <= n; i++)
for(int j = ; j <= m; j++) {
read(a[i][j]);
sum[i][j] = sum[i][j - ] + a[i][j];
}
for(int i = ; i <= n; i++)
for(int j = ; j <= m; j++)
sum[i][j] += sum[i - ][j];
for(int i = k; i <= n; i++)
for(int j = k; j <= m; j++)
s[i][j] = sum[i][j] + sum[i - k][j - k] - sum[i][j - k] - sum[i - k][j]; /* for(int i = 1; i <= n; i++, printf("\n"))
for(int j = 1; j <= m; j++)
printf("%d ", s[i][j]); */ for(int i = ; i <= n; i++)
for(int j = ; j <= m; j++)
chkMax(f1[i][j], max(s[i][j], f1[i - ][j], f1[i][j - ]));
for(int i = ; i <= n; i++)
for(int j = m; j >= ; j--)
chkMax(f2[i][j], max(s[i][j + k - ], f2[i - ][j], f2[i][j + ]));
for(int i = n; i >= ; i--)
for(int j = m; j >= ; j--)
chkMax(f3[i][j], max(s[i + k - ][j + k - ], f3[i + ][j], f3[i][j + ]));
for(int i = n; i >= ; i--)
for(int j = ; j <= m; j++)
chkMax(f4[i][j], max(s[i + k - ][j], f4[i + ][j], f4[i][j - ])); ans = ;
for(int i = k; i <= n - k; i++)
for(int j = k; j <= m - k; j++) {
chkMax(ans, f1[i][j] + f2[i][j + ] + f3[i + ][]);
chkMax(ans, f1[i][j] + f2[n][j + ] + f4[i + ][j]);
chkMax(ans, f1[n][j] + f2[i][j + ] + f3[i + ][j + ]);
chkMax(ans, f1[i][m] + f4[i + ][j] + f3[i + ][j + ]);
} for(int i = ; i <= n; i++)
for(int j = ; j <= m; j++) {
chkMax(r[i], s[i][j]);
chkMax(c[j], s[i][j]);
} for(int i = k; i <= n - * k; i++)
for(int j = i + k, mid = r[j]; j <= n - k; j++, chkMax(mid, r[j]))
chkMax(ans, f1[i][m] + mid + f3[j + ][]);
for(int i = k; i <= m - * k; i++)
for(int j = i + k, mid = c[j]; j <= m - k; j++, chkMax(mid, c[j]))
chkMax(ans, f1[n][i] + mid + f2[n][j + ]); printf("%d\n", ans);
return ;
}

Luogu 3625 [APIO2009]采油区域的更多相关文章

  1. [APIO2009]采油区域

    题目描述 Siruseri 政府决定将石油资源丰富的 Navalur 省的土地拍卖给私人承包商以 建立油井.被拍卖的整块土地为一个矩形区域,被划分为 M×N 个小块. Siruseri 地质调查局有关 ...

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

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

  3. [SOJ #686]抢救(2019-11-7考试)/[洛谷P3625][APIO2009]采油区域

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

  4. [P3625][APIO2009]采油区域 (前缀和)

    这道题用二维前缀和可以做 难度还不算高,细节需要注意 调试了很久…… 主要是细节太多了 #include<bits/stdc++.h> using namespace std; #defi ...

  5. bzoj1177&p3625 [APIO2009]采油区域p[大力讨论]

    我好菜菜啊. 给定矩形,从中选出三个边长K的正方形互不重叠,使得覆盖到的数总和最大. 想的时候往dp上钻去了..结果一开始想了一个错的dp,像这样 /************************* ...

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

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

  7. Java实现 蓝桥杯VIP 算法训练 采油区域

    算法训练 采油区域 时间限制:2.0s 内存限制:512.0MB 提交此题 查看参考代码 采油区域 Siruseri政府决定将石油资源丰富的Navalur省的土地拍卖给私人承包商以建立油井.被拍卖的整 ...

  8. 【luogu P3627 [APIO2009]抢掠计划】 题解

    题目链接:https://www.luogu.org/problemnew/show/P3627 把点权转化到边权上去. #include <stack> #include <que ...

  9. [Luogu P3626] [APIO2009] 会议中心

    题面 传送门:https://www.luogu.org/problemnew/show/P3626 Solution 如果题目只要求求出第一问,那这题显然就是大水题. 但是加上第二问的话...... ...

随机推荐

  1. GUID在.net里的使用

    GUID(全局统一标识符)是指在一台机器上生成的数字,它保证对在同一时空中的所有机器都是唯一的.通常平台会提供生成GUID的API.生成算法很有意思,用到了以太网卡地址.纳秒级时间.芯片ID码和许多可 ...

  2. Effective C++ 条款10

    令operator=返回一个reference to *this 将operator=返回一个reference是为了什么呢?答案很简单,就是为了实现连锁形式. 什么是连锁形式,如int x,y,z: ...

  3. 使用open_read_write等底层函数来赋值一个文件

    /* * 该程序是练习read(),write(),open(),create(),close(),lseek()等函数. *  * 该程序的处理思路: *  1: 在程序所在的目录去打开一个文件,如 ...

  4. Technocup 2019 C. Compress String

    一个字符串 $s$,你要把它分成若干段,有两种合法的段 1.段长为 $1$,代价为 $a$ 2.这个段是前面所有段拼起来组成的字符串的字串,代价为 $b$ 问最小代价 $|s| \leq 5000$ ...

  5. Cash Machine(多重背包二进制转换)

    个人心得:多重背包,自己根据转换方程写总是TLE,后面去网上看了二进制转换,不太理解: 后面仔细想了下,用自己的思想理解下把,就是将对应number,cash总和用二进制拆分, 然后全部装入到一个数组 ...

  6. LeetCode Target Sum

    原题链接在这里:https://leetcode.com/problems/target-sum/description/ 题目: You are given a list of non-negati ...

  7. java实现sendemail

    <dependency> <groupId>com.sun.mail</groupId> <artifactId>javax.mail</arti ...

  8. redis之 centos 6.7 下安装 redis-3.2.5

    前期准备:1. 操作系统需要安装 gcc 包 与  TCL 库, 通过配置本地 yum 源 ,yum -y install gcc . yum -y install tcl安装2. 下载 redis ...

  9. Operating System-进程/线程内部通信-信号量和PV操作

    本文介绍操作系统进程管理的两个核心概念: 信号量 PV操作 一.信号量介绍 1.1 信号量引入 信号量(Semaphore)1965年由Dijkstra引入的.信号量一般由一个值是一个变量,其值有可能 ...

  10. why latches are considered bad?

    A "latch" is different from a "Flip-Flop" in that a FF only changes its output i ...