[Codeforces 321D][2018HN省队集训D4T2] Ciel and Flipboard

题意

给定一个 \(n\times n\) 的矩阵 \(A\), (\(n\) 为奇数) , 每次可以选 \(A\) 的一个 \(\frac {n+1}2 \times \frac {n+1} 2\) 的子矩阵并让这个子矩阵中的所有值取反.

进行若干次操作最大化整个矩阵中的元素值之和. 输出这个最大值.

\(n\le 33\), \(|A_{i,j}|\le 1000\)

题解

毒瘤wls活该被A

hzoi2017_jjm 当场AC, 大强辣!

这题是个结论题.

首先我们看他 \(n\le 33\) 必有高论. 实际上就是个结论优化暴力.

接着我们发现这个 \(\frac{n+1}2\) 非常奥妙重重. 设这个值为 \(m\). 它刚好卡在比一半稍多的位置, 中间的一行一列经常被翻转. 或者说, 只要 \((i,j)\) 被翻转, \((i,m)\) 和 \((m,j)\) 一定也被翻了. 如果 \((i,j)\) 没被翻但是 \((i,m)\) 被翻了, 那么肯定当前操作的子矩阵就被怼到一边去, 导致 \((i,j\pm m)\) 被翻. 不难发现 \((i,j),(i,m),(i,j+m)\) 三个位置在一次操作中如果有一个被翻, 那么必定有且仅有另一个被翻. 也就是说这三个位置的翻转状态的异或和不变且一直是 \(0\).

这个结论显然对于另一维也成立. \((i,j),(m,j),(i+m,j)\) 三个位置的翻转状态的异或和也是 \(0\).

这三个位置的翻转状态只要知道两个显然就能计算出第三个. 而这些关系都和 \((i,m)\) 以及 \((m,j)\) 有关. 我们考虑枚举这些用得很多的位置的翻转状态. (注意到我们对于第 \(m\) 行/列, 只需要枚举一半就可以推出另一半的状态.) 容易发现第 \(m\) 行和第 \(m\) 列的状态确定后, 剩余的位置被分为若干形如 \(\{(i,j),(i+m,j),(i,j+m),(i+m,j+m)\}\) 的组合, 组合之间互相不再有影响. 于是我们可以枚举其中一个位置的状态推出其余位置的状态, 然后两种情况取 \(\max\) 求和即为答案.

虽然我们只需要枚举一半, 但是总枚举量还是有 \(2^n=2^{33}\approx 8\times 10^9\). 再加上还需要 \(O(n^2)\) 验证显然非常不靠谱.

我们又惊奇地发现, 枚举行之后, \(\{(i,j),(i+m,j),(i,j+m),(i+m,j+m)\}\) 只和 \((i,m)\) 有关. 于是我们可以分别枚举 \((i,m)\) 的状态计算一遍和再取 \(\max\) 最后求和.

总时间复杂度 \(O(2^mn^2)\).

参考代码

#include <bits/stdc++.h>

const int MAXN=50;
const int k[2]={1,-1}; int n;
int a[MAXN][MAXN];
int d[MAXN][MAXN]; int main(){
scanf("%d",&n);
for(int i=0;i<n;i++)
for(int j=0;j<n;j++)
scanf("%d",a[i]+j);
int m=(n+1)>>1;
int ans=INT_MIN;
for(int s=0;s<(1<<m);s++){
int sum=0;
for(int i=0;i<m;i++)
d[m-1][i]=((s>>i)&1)?1:-1;
for(int i=m;i<n;i++)
d[m-1][i]=d[m-1][i-m]*d[m-1][m-1];
for(int i=0;i<n;i++)
sum+=d[m-1][i]*a[m-1][i];
for(int i=0;i<m-1;i++){
int cur=INT_MIN;
for(int r=0;r<2;r++){
d[i][m-1]=k[r];
d[i+m][m-1]=d[i][m-1]*d[m-1][m-1];
int now=d[i][m-1]*a[i][m-1]+d[i+m][m-1]*a[i+m][m-1];
for(int j=0;j<m-1;j++){
int tmp=INT_MIN;
for(int r=0;r<2;r++){
d[i][j]=k[r];
d[i+m][j]=d[i][j]*d[m-1][j];
d[i][j+m]=d[i][j]*d[i][m-1];
d[i+m][j+m]=d[i+m][j]*d[i+m][m-1];
tmp=std::max(tmp,d[i][j]*a[i][j]+d[i+m][j]*a[i+m][j]+d[i][j+m]*a[i][j+m]+d[i+m][j+m]*a[i+m][j+m]);
}
now+=tmp;
}
cur=std::max(cur,now);
}
sum+=cur;
}
ans=std::max(ans,sum);
}
printf("%d\n",ans);
return 0;
}

[Codeforces 321D][2018HN省队集训D4T2] Ciel and Flipboard的更多相关文章

  1. [2018HN省队集训D9T1] circle

    [2018HN省队集训D9T1] circle 题意 给定一个 \(n\) 个点的竞赛图并在其中钦定了 \(k\) 个点, 数据保证删去钦定的 \(k\) 个点后这个图没有环. 问在不删去钦定的这 \ ...

  2. [2018HN省队集训D8T1] 杀毒软件

    [2018HN省队集训D8T1] 杀毒软件 题意 给定一个 \(m\) 个01串的字典以及一个长度为 \(n\) 的 01? 序列. 对这个序列进行 \(q\) 次操作, 修改某个位置的字符情况以及查 ...

  3. [2018HN省队集训D8T3] 水果拼盘

    [2018HN省队集训D8T3] 水果拼盘 题意 给定 \(n\) 个集合, 每个集合包含 \([1,m]\) 中的一些整数, 在这些集合中随机选取 \(k\) 个集合, 求这 \(k\) 个集合的并 ...

  4. [2018HN省队集训D6T2] girls

    [2018HN省队集训D6T2] girls 题意 给定一张 \(n\) 个点 \(m\) 条边的无向图, 求选三个不同结点并使它们两两不邻接的所有方案的权值和 \(\bmod 2^{64}\) 的值 ...

  5. [Luogu P4143] 采集矿石 [2018HN省队集训D5T3] 望乡台platform

    [Luogu P4143] 采集矿石 [2018HN省队集训D5T3] 望乡台platform 题意 给定一个小写字母构成的字符串, 每个字符有一个非负权值. 输出所有满足权值和等于这个子串在所有本质 ...

  6. [2018HN省队集训D5T2] party

    [2018HN省队集训D5T2] party 题意 给定一棵 \(n\) 个点以 \(1\) 为根的有根树, 每个点有一个 \([1,m]\) 的权值. 有 \(q\) 个查询, 每次给定一个大小为 ...

  7. [2018HN省队集训D5T1] 沼泽地marshland

    [2018HN省队集训D5T1] 沼泽地marshland 题意 给定一张 \(n\times n\) 的棋盘, 对于位置 \((x,y)\), 若 \(x+y\) 为奇数则可能有一个正权值. 你可以 ...

  8. [2018HN省队集训D1T3] Or

    [2018HN省队集训D1T3] Or 题意 给定 \(n\) 和 \(k\), 求长度为 \(n\) 的满足下列条件的数列的数量模 \(998244353\) 的值: 所有值在 \([1,2^k)\ ...

  9. [2018HN省队集训D1T1] Tree

    [2018HN省队集训D1T1] Tree 题意 给定一棵带点权树, 要求支持下面三种操作: 1 root 将 root 设为根. 2 u v d 将以 \(\operatorname{LCA} (u ...

随机推荐

  1. ARM的体系结构与编程系列博客——ARM的历史与应用范围

    前言 最近我感觉自己比较浮躁,重来没有好好地沉下心来做一件事情,而且针对自己在专业水平上仍然还有很多欠缺,于是我想我应该为自己做些什么来证明一下自己真的是潜心研究东西的人,于是我萌生了一个想法,真正地 ...

  2. For update带来的思考

    For update or not 起源 ​ 之所以想写这个专题,是因为最近在做一个抢占任务的实现.假设数据库很多个任务,在抢占发生之前任务的状态都是FREE.现在假设同时有一堆抢占线程开始工作,抢占 ...

  3. gRPC 的route_guide例子

      本文的例子代码在: https://github.com/grpc/grpc-go/tree/master/examples/route_guide 功能就类似目前LBS一样,在每个位置上报一些文 ...

  4. Notepad++怎么使用正则替换

    前言:工作中在oracle中写触发器的sql时,表字段有几十个,修改起来非常不方便,于是采用了Notepad++的替换 案例: 想把 v_update_time,v_create_time,v_rcv ...

  5. c#如何判断两个对象是否相等

    在c#中判断对象相等,这是对引用类型进行判断,而不是对值类型,如果是对字符串,或者是数值进行判断相等只需要用==运算符就可以了. 对两个对象用==运算符,只能判断他们两个在内存中的地址是否一样的.   ...

  6. 转:导出csv文件数字会自动变科学计数法的解决方法

    导出csv文件数字会自动变科学计数法的解决方法   其实这个问题跟用什么语言导出csv文件没有关系.Excel显示数字时,如果数字大于12位,它会自动转化为科学计数法:如果数字大于15位,它不仅用于科 ...

  7. 纯css修改复选框默认样式

    input[type='checkbox']{ width: 20px; height: 20px; background-color: #fff; -webkit-appearance:none; ...

  8. 当div元素内的内容超出其宽度时,自动隐藏超出的内容

    word-break:keep-all;/* 不换行 */ white-space:nowrap;/* 不换行 */ overflow:hidden;/* 内容超出宽度时隐藏超出部分的内容 */ te ...

  9. 如何配置Portal 基于AD的单点登录配置

    Portal for ArcGIS支持两种类型的账户,分别是: 1.系统内置账户. 2.外部系统的企业账户. 这两种不同的账号分别支持多种身份认证方式: 账号类型 认证方式 细分认证方式 系统内置账号 ...

  10. Pig安装与应用

    1.  参考说明 参考文档: http://pig.apache.org/docs/r0.17.0/start.html#build 2.  安装环境说明 2.1.  环境说明 CentOS7.4+ ...