CF-1440C2 Binary Table (Hard Version) (构造,模拟)
Binary Table (Hard Version)
题意
\(n*m(2\le n,m\le 100)\) 的01矩阵,每次可以选择一个宽度为2的子矩阵,将四个位置中的任意3个进行翻转,即0变1,1变0。要求构造操作次数小于 \(n*m\) 的方案,使得该矩阵最终变成一个全0矩阵。
分析
构造方法可能有很多种,只要能够满足题意即可。
从第一行开始到倒数第三行,逐列扫描,假设当前扫描的位置为 (i,j)
,且当前位置为1,那么执行操作一或者操作二。\(i\le n-2\) ,只扫描前 \(n-2\) 行,最后空下两行特殊处理。这之前的 \((n-2)*m\) 个位置,总共使用了不超过 \((n-2)*m\) 次操作。
处理最后两行时,需要4个一组进行处理。分情况讨论
- 如果 4 个都是 ‘1’ ,进行一次翻转,只留右下角的 1,这个 1 留到最后处理
- 如果有某 3 个是 ‘1’ ,那么一次操作可以搞定
- 如果有某 2 个是 '1',那么可以细分为两种情况,这两种情况都可以用两步解决
- 如果只有 1 个是 '1',那么留作最后考虑。
根据上面的情况,对最后两行进行处理,如果 \(m\) 是偶数,刚好可以处理完。但如果 \(m\) 是奇数,那么对于最后一列的两个格子,我们需要特殊考虑。
可以想到,我们按照 4 个一组处理完左边四个之后,最多只会在 (n,m-1)
这个位置留一个 1,那么不论第 m 列是什么情况,我们都可以在两步操作内,将他们变为0。这个操作是绝对保险的。
最后,只剩下最后两行的某些单独的 1 了,这些单独的 1 可以在 3 步内消除,至于为什么这么操作不会使得操作次数爆炸,可以从这些 1 的产生来源考虑。
这个 1 原本就摆在这里,那么旁边 3 个空位之前是没有操作过的,这就给这个 1 留下了多余的操作空间。
这个 1 是消了 3 个之后剩下的,那么消除那 3 个用了一次,消除单独的 1 用了 3 次,刚好四次。
对于 m 为奇数时,最右边一列中单独的 1,这个虽然也需要使用 3 步,但是由于此时 m-1 列都是空的,这得多亏左边那四个之前是消除干净了的。而 4 个一组消除干净最多只需要两步,所以这种情况也是保险的。
#include <bits/stdc++.h>
using namespace std;
const int N = 101;
int T, n, m, t[N][N];
char s[N][N];
vector<pair<int,int> > v;
void add(int x, int y){
v.push_back({x, y});
t[x][y] ^= 1;
}
void add(int x, int y, int p){
if(p == 0) add(x, y);
else if(p == 1) add(x, y+1);
else if(p == 2) add(x+1, y);
else if(p == 3) add(x+1, y+1);
}
void solve(int x, int y){
int c0 = t[x][y];
int c1 = t[x][y+1];
int c2 = t[x+1][y];
int c3 = t[x+1][y+1];
int c = c0 + c1 + c2 + c3;
if(c == 4) {
add(x, y, 0);
add(x, y, 1);
add(x, y, 2);
} else if(c == 3) {
if(c0 == 0) add(x, y, 1), add(x, y, 2), add(x, y, 3);
if(c1 == 0) add(x, y, 0), add(x, y, 2), add(x, y, 3);
if(c2 == 0) add(x, y, 0), add(x, y, 1), add(x, y, 3);
if(c3 == 0) add(x, y, 0), add(x, y, 1), add(x, y, 2);
} else if(c == 2){
if(c0 && c1) {
add(x, y, 0);add(x, y, 2);add(x, y, 3);
add(x, y, 1);add(x, y, 2);add(x, y, 3);
} else if(c0 && c2) {
add(x, y, 0);add(x, y, 1);add(x, y, 3);
add(x, y, 2);add(x, y, 1);add(x, y, 3);
} else if(c0 && c3) {
add(x, y, 0);add(x, y, 1);add(x, y, 2);
add(x, y, 3);add(x, y, 1);add(x, y, 2);
} else if(c1 && c2){
add(x, y, 1);add(x, y, 0);add(x, y, 3);
add(x, y, 2);add(x, y, 0);add(x, y, 3);
} else if(c1 && c3) {
add(x, y, 1);add(x, y, 0);add(x, y, 2);
add(x, y, 3);add(x, y, 0);add(x, y, 2);
} else if(c2 && c3) {
add(x, y, 2);add(x, y, 0);add(x, y, 1);
add(x, y, 3);add(x, y, 0);add(x, y, 1);
}
}
}
int main(){
scanf("%d", &T);
while(T--){
scanf("%d%d", &n, &m);
for(int i=1;i<=n;i++) scanf("%s", s[i]+1);
for(int i=1;i<=n;i++){
for(int j=1;j<=m;j++){
t[i][j] = s[i][j] - '0';
}
}
for(int i=1;i<=n-2;i++){
for(int j=1;j<=m;j++){
if(t[i][j] == 0) continue;
add(i, j); add(i+1, j);
if(j == m) add(i+1, j-1);
else add(i, j+1);
}
}
for(int j = 1; j + 1 <= m; j += 2){
solve(n - 1, j);
}
if(m & 1) solve(n - 1, m - 1);
// 处理倒数第二行中单着的
for(int j = 1; j <= m; j++){
if(t[n-1][j] == 0) continue;
if(j == m) {
add(n-1, j-1, 1); add(n-1, j-1, 0); add(n-1, j-1, 3);
add(n-1, j-1, 0); add(n-1, j-1, 2); add(n-1, j-1, 1);
add(n-1, j-1, 3); add(n-1, j-1, 2); add(n-1, j-1, 1);
} else {
add(n-1, j, 0); add(n-1, j, 1); add(n-1, j, 2);
add(n-1, j, 2); add(n-1, j, 0); add(n-1, j, 3);
add(n-1, j, 1); add(n-1, j, 0); add(n-1, j, 3);
}
}
// 处理倒数第一行单着的
for(int j = 1; j <= m; j++){
if(t[n][j] == 0) continue;
if(j == m) {
add(n-1, j-1, 3); add(n-1, j-1, 2); add(n-1, j-1, 1);
add(n-1, j-1, 2); add(n-1, j-1, 0); add(n-1, j-1, 3);
add(n-1, j-1, 1); add(n-1, j-1, 0); add(n-1, j-1, 3);
} else {
add(n-1, j, 2); add(n-1, j, 0); add(n-1, j, 3);
add(n-1, j, 0); add(n-1, j, 1); add(n-1, j, 2);
add(n-1, j, 3); add(n-1, j, 1); add(n-1, j, 2);
}
}
cout << v.size() / 3 << endl;
for(int i=0;i<v.size();){
#define x first
#define y second
printf("%d %d %d %d %d %d\n", v[i].x, v[i].y, v[i+1].x, v[i+1].y, v[i+2].x, v[i+2].y);
i += 3;
}
v.clear();
}
return 0;
}
CF-1440C2 Binary Table (Hard Version) (构造,模拟)的更多相关文章
- CF 1003B Binary String Constructing 【构造/找规律/分类讨论】
You are given three integers a, b and x. Your task is to construct a binary string s of length n=a+b ...
- CF 662C Binary Table
用FWT优化计算. 首先发现行数很小,想到一个暴力的方法,就是以一个二进制位$0$表示这一行不翻转而二进制位$1$表示这一行翻转,然后$2^n$枚举出所有行的翻转情况,再$O(m)$计算所有的结果. ...
- D2. Kirk and a Binary String (hard version) D1 Kirk and a Binary String (easy version) Codeforces Round #581 (Div. 2) (实现,构造)
D2. Kirk and a Binary String (hard version) time limit per test1 second memory limit per test256 meg ...
- 【CF662C】Binary Table(FWT)
[CF662C]Binary Table(FWT) 题面 洛谷 CF 翻译: 有一个\(n*m\)的表格(\(n<=20,m<=10^5\)), 每个表格里面有一个\(0/1\), 每次可 ...
- [CF662C Binary Table][状压+FWT]
CF662C Binary Table 一道 FWT 的板子-比较难想就是了 有一个 \(n\) 行 \(m\) 列的表格,每个元素都是 \(0/1\),每次操作可以选择一行或一列,把 \(0/1\) ...
- CROC 2016 - Final Round [Private, For Onsite Finalists Only] C. Binary Table FWT
C. Binary Table 题目连接: http://codeforces.com/problemset/problem/662/C Description You are given a tab ...
- 【CF662C】Binary Table 按位处理
[CF662C]Binary Table 题意:给你一个$n\times m$的01网格,你可以进行任意次操作,每次操作是将一行或一列的数都取反,问你最多可以得到多少个1? $n\le 20,m\le ...
- D1. Kirk and a Binary String (easy version)
D1. Kirk and a Binary String (easy version) 01串找最长不降子序列 给定字符串s,要求生成一个等长字符串t,使得任意l到r位置的最长不降子序列长度一致 从后 ...
- CF662C Binary Table【FWT】
CF662C Binary Table 题意: 给出一个\(n\times m\)的\(01\)矩阵,每次可以反转一行或者一列,问经过若干次反转之后,最少有多少个\(1\) \(n\le 20, m\ ...
随机推荐
- LeetCode682 棒球比赛
题目描述: 你现在是棒球比赛记录员.给定一个字符串列表,每个字符串可以是以下四种类型之一:1.整数(一轮的得分):直接表示您在本轮中获得的积分数.2. "+"(一轮的得分):表示本 ...
- 你必须要懂的 Github 开源协议
作为一个开源社区的活跃者,那些开源协议你都懂什么意思吗? 列两个: Apache License 可以: 商用.修改.分发 但是要声明作者来源和你的修改以及协议 MIT License 只要声明版权 ...
- 【Linux】记一次xfs分区数据恢复
项目有一块磁盘无法挂载,而且还没有做RAID.... # mount /dev/sda /xxx 报错 mount: special device /dev/sda/ does not exist ...
- kubernets之job资源
一 介绍job资源 1.1 前面介绍的RC,RS,DS等等,管控的pod都是需要长期持久的运行的应用,但是尝试考虑另外一种场景,在微服务的场景下,有些pod的作用就是需要 执行完一些命令之后正常 ...
- 洛谷P1198 [JSOI2008]最大数(线段树/单调栈)
题目链接: https://www.luogu.org/problemnew/show/P1198 题目描述 现在请求你维护一个数列,要求提供以下两种操作: 1. 查询操作. 语法:Q L 功能:查询 ...
- Android之Xposed
基础书籍推荐:1.疯狂JAVA讲义:2.疯狂安卓讲义: 逆向分析必须知道他的原理,不然只会用工具,那就直接GG. 谷歌的镜像网站:https://developers.google.com/andro ...
- 关于JDK15的简单理解
一.为什么要了解JDK15? 2020年9月15日,Oracle官方发布了JDK15版本,及时关注官方的更新动态,可以让我们在日常开发中更合理的选择更加优秀的工具方法,避免使用一些过时的或一些即将被删 ...
- 【Not BUG】微软Winform窗体中设计上的Bug,会导致程序编译失败?不,这不是BUG!
这不是BUG!!! 原文地址: https://www.cnblogs.com/thanks/p/14302011.html 现在让我们回忆一下原文 原文的操作步骤: 1. 新建一个Window Fo ...
- [USACO13DEC]牛奶调度Milk Scheduling
原题链接https://www.lydsy.com/JudgeOnline/problem.php?id=4096 容易想到的一个测略就是,优先考虑结束时间小的牛.所以我们对所有牛按照结束时间排序.然 ...
- 显示HDFS中指定的文件读写权限、大小、创建时间、路径等信息。
1 import org.apache.hadoop.fs.*; 2 import java.text.SimpleDateFormat; 3 public class D_ReadFileStatu ...