WJQ与机房

sample input
5 6
7 2 3 1 1
5 0 6 0 0
8 6 6 5 3
4 3 7 8 2
4 0 0 6 9
sample output
20
样例解释:
分别以(2,1)为左上角(5,2)为右下角和以(1,3)为左上角以(4,5)
为右下角的两个矩阵。
数据范围:
对于20的数据n≤5
对于50的数据n≤50
另有5的数据满足所有电脑的便利值均为1
另另有25的数据满足所有的电脑便利值相同
对于100的数据满足n≤300,p≤50,0≤a[i][j]≤65536
首先,我们会想到n^4的暴力dp,但一看n的范围,完全装不下。
所以我们要考虑优化掉一维。
我们考虑枚举l和r这两条线,代表矩形的左右边界,同时我们对每一行求一个前缀和,方便我们更新。

l和r为矩形的左右两边,i和tong[i]为上下两边。
tong[i]的作用下次再说。
这样,我们就会得到每个矩形的和,并考虑对答案的贡献。
我们开两个数组f[i],g[i]
g[i]表示i这条直线的右边的最大矩形(不一定紧贴)
f[i]是紧贴着i这条线的左边的最大矩形
tong[i]存每一次矩形数字之和%p后的结果,第一次出现在哪一行。-1为没有出现过。(可能有点绕口雾)
tot为当前矩形的数字之和 tot += sum[i][r] - sum[i][l-1];
设k为tot % p之后的结果,当k不为1时会有以下两种情况。
1.k为0,是p的倍数,表明可以更新答案。
2.k这个结果以前出现过,那么tong[k]到k这一行矩形的数字和也恰好是p的倍数。
我们就可以更新一下两个数组。。
f[r] = max(f[r],(r-l+1) * (i-tong[tot]));
g[l] = max(g[l],(r-l+1) * (i-tong[tot]));
当我们矩形都找完后,还要在更新一下g数组,因为我们只考虑了紧贴的情况,而没有考虑不紧贴。
务必要倒叙枚举,因为后面的答案可以影响前面的结果,所以要倒序。
那么答案是什么呢???
for(int i = 1; i < n-1; i++) ans = max(ans,f[i] + g[i+1]);
i一定要到n-2不能到n-1,因为到n的话g数组的定义就会矛盾。
g[i+1]是为了防止与f[i]的边重叠。
PS:
对于为什么在“第二种:以前出现过模p得到k的情况”的时候也要更新答案?“
我们重新考虑题意:一块矩形的和是p的倍数,也就是模p为零
可以用式子表示为:
s当前−s上一次≡0(modp)
进一步转化为:
s当前≡s上一次(modp)
所以可理解为,只要上一次和这一次在模p意义的数字一样,就可以更新答案了,因为他们相减之后一定是p的倍数,可手跑几组例子
至此,关于“第二种:以前出现过模p得到k的情况”的时候也要更新答案?”这句话,也就不难理解了——from Thestars。
但我们发现少考虑了一种一个矩形在上,一个矩形在下的情况,这是我们只要旋转一下,在跑一遍就ok了
代码
#include<iostream>
#include<cstdio>
#include<algorithm>
using namespace std;
int n,p,f[310],g[310],sum[310][310],tong[55],a[310][310],ans;
inline int read()
{
int s = 0, w = 1; char ch = getchar();
while(ch < '0' || ch > '9'){if(ch == '-') w = -1; ch = getchar();}
while(ch >= '0' && ch <= '9'){s = s * 10+ch -'0'; ch = getchar();}
return s * w;
}
void slove()
{
for(int i = 1; i <= n; i++) f[i] = g[i] = 0;
for(int i = 1; i <= n; i++)
{
for(int j = 1; j <= n; j++)
{
sum[i][j] = sum[i][j-1] + a[i][j];//每一行求一个区间和
}
}
for(int l = 1; l <= n; l++)//枚举左边的线
{
for(int r = l; r <= n; r++)//右边的线
{
for(int i = 1; i <= p ;i++) tong[i] = -1;//桶里存模p结果第一次出现的行数 ,tong[0] 不要忘记初始化为0
int tot = 0;
for(int i = 1; i <= n; i++)
{
tot += sum[i][r] - sum[i][l-1];
tot %= p;//模p的结果开个桶
if(tong[tot] != -1)//不是第一次出现
{
f[r] = max(f[r],(r-l+1) * (i-tong[tot]));//更新
g[l] = max(g[l],(r-l+1) * (i-tong[tot]));
}
else tong[tot] = i;
}
}
}
for(int i = n-1; i; i--) g[i] = max(g[i],g[i+1]);//这一步考虑的是没有紧贴的情况
for(int i = 1; i < n-1; i++) ans = max(ans,f[i] + g[i+1]);
}
void zhuan()
{
for(int i = 1; i <= n; i++)
{
for(int j = i+1; j <= n; j++)
{
int xx = a[i][j]; int yy = a[j][i];
xx ^= yy; yy ^= xx; xx ^= yy;
a[i][j] = xx; a[j][i] = yy;
}
}
}
int main()
{
n = read(); p = read();
for(int i = 1; i <= n; i++)
{
for(int j = 1; j <= n; j++)
{
a[i][j] = read();
}
}
slove();
zhuan();
slove();
printf("%d\n",ans);
fclose(stdin); fclose(stdout);
return 0;
}
ENDING
WJQ与机房的更多相关文章
- 最近帮客户实施的基于SQL Server AlwaysOn跨机房切换项目
最近帮客户实施的基于SQL Server AlwaysOn跨机房切换项目 最近一个来自重庆的客户找到走起君,客户的业务是做移动互联网支付,是微信支付收单渠道合作伙伴,数据库里存储的是支付流水和交易流水 ...
- 从0开始搭建SQL Server AlwaysOn 第四篇(配置异地机房节点)
从0开始搭建SQL Server AlwaysOn 第四篇(配置异地机房节点) 第一篇http://www.cnblogs.com/lyhabc/p/4678330.html第二篇http://www ...
- SQL Server 跨网段(跨机房)FTP复制
一.本文所涉及的内容(Contents) 本文所涉及的内容(Contents) 背景(Contexts) 搭建过程(Process) 注意事项(Attention) 参考文献(References) ...
- SQL Server 跨网段(跨机房)复制
一.本文所涉及的内容(Contents) 本文所涉及的内容(Contents) 背景(Contexts) 解决方案(Solution) 搭建过程(Process) 注意事项(Attention) 参考 ...
- IT基础架构规划方案二(计算机系统与机房规划规划)
计算机系统规划 服务器硬件选型规划方案 根据对某集团的实际调研,获取了企业业务应用系统的建设情况,随着企业信息化建设的推进,需要对各种信息化管理系统和应用系统的服务器选型进行选 ...
- IDC机房线路质量测试方案
1.测试节点: 北京:123.206.*.* 上海:139.196.*.* 广州:119.29.*.* 汕头:125.91.*.* 香港:103.20.*.* 美国:198.52.*.* 测试服务器: ...
- Codevs 2370 小机房的树 LCA 树上倍增
题目描述 Description 小机房有棵焕狗种的树,树上有N个节点,节点标号为0到N-1,有两只虫子名叫飘狗和大吉狗,分居在两个不同的节点上.有一天,他们想爬到一个节点上去搞基,但是作为两只虫子, ...
- [转]搬瓦工换机房换ip之后不能连外网
搬瓦工换机房换ip之后不能连外网 时间 2015-07-21 15:17:16 Wendal随笔 原文 http://wendal.net/2015/07/21.html 主题 iptables ...
- 本机,同机房,同城,异地,不同城,腾讯云ping延时值
本机,同机房,同城,异地,不同城,腾讯云ping延时值 ping本机: 0.01ms ping同机房机器: 0.1ms ping同城机器: 1ms ping不同城机器: 20ms 北(南)方ping南 ...
随机推荐
- 寻找链表的倒数第k个节点
寻找链表的倒数第k个节点 题目:已知一个带有表头结点的单链表,节点结构为(data,next),假设该链表只给出了头指针list.在不改变链表的前提下,请设计一个尽可能高效的算法,查找链表中倒数第k个 ...
- 最强 Java 书单推荐,附学习方法
技术大佬用1w+字来告诉你该读什么书,循序渐进,并提供百度云盘下载地址.重要的是还有学习方法. 请肆无忌惮地点赞吧,微信搜索[沉默王二]关注这个在九朝古都洛阳苟且偷生的程序员.本文 GitHub gi ...
- 容器服务 TKE 上服务暴露的几种方式
预备知识 1. K8S 上 Service 类型 ClusterIP 通过集群的内部 IP 暴露服务,选择该值,服务只能够在集群内部可以访问,这也是默认的 ServiceType. NodePort ...
- java 多线程-2
七.线程生命周期 没错,线程也是有生命周期的.就好像人类有出生.儿童.青年.中年.晚年.死亡一般.下面是线程的生命周期图: 八.线程的安全问题 所谓线程不安全[并发问题],举个例子来说,如卖票,会出现 ...
- Java单例模式的实现与破坏
单例模式是一种设计模式,是在整个运行过程中只需要产生一个实例.那么怎样去创建呢,以下提供了几种方案. 一.创建单例对象 懒汉式 public class TestSingleton { // 构造方法 ...
- 为什么互联网巨头们纷纷使用Git而放弃SVN?(内含Git核心命令与原理总结)
写在前面 最近发现很多小伙伴对于工作中的一些基本工具的使用还不是很了解,比如:Git这个分布式的代码管理仓库,很多小伙伴就不是很了解,或者说不是很熟悉.甚至有些小伙伴都没听说过Git,就只会用个SVN ...
- xss原理解析
xss->跨站脚本攻击 xss是指攻击者在网页中嵌入客户端脚本.通常是指javascript编写的一个危险代码,当用户使用浏览器浏览网页时,脚本就会在用户的浏览器上执行,从而达到攻击者的目的. ...
- [Leetcode]225. 用队列实现栈 、剑指 Offer 09. 用两个栈实现队列
##225. 用队列实现栈 如题 ###题解 在push时候搞点事情:push时入队1,在把队2的元素一个个入队1,再交换队2和队1,保持队1除pushguocheng 始终为空. ###代码 cla ...
- hystrix熔断器之线程池
隔离 Hystrix有两种隔离方式:信号量和线程池. 线程池隔离:对每个command创建一个自己的线程池,执行调用.通过线程池隔离来保证不同调用不会相互干扰和每一个调用的并发限制. 信号量隔热:对每 ...
- leetcode560题解【前缀和+哈希】
leetcode560.和为K的子数组 题目链接 算法 前缀和+哈希 时间复杂度O(n). 在解决这道题前需要先清楚,一个和为k的子数组即为一对前缀和的差值. 1.我们假设有这么一个子数组[i,j]满 ...