NC235250 牛可乐的翻转游戏
NC235250 牛可乐的翻转游戏
题目
题目描述
牛可乐发明了一种新型的翻转游戏!
在一个有 \(n\) 行 \(m\) 列的棋盘上,每个格子摆放有一枚棋子,每一枚棋子的颜色要么是黑色,要么是白色。每次操作牛可乐可以选择一枚棋子,将它的颜色翻转(黑变白,白变黑),同时将这枚棋子上下左右相邻的四枚棋子的颜色翻转(如果对应位置有棋子的话)。
牛可乐想请你帮他判断一下,能否通过多次操作将所有棋子都变成黑色或者白色?如果可以,最小操作次数又是多少呢?
输入描述
第一行两个整数 \(n,m(1\leq n\leq 100,1\leq m\leq 10)\),代表棋盘的行数和列数。
之后 \(n\) 行,每行 \(m\) 个数字,第 \(i\) 个数字如果为 \(0\) ,代表对应位置的棋子为白色,如果为 \(1\) 则为黑色。
输出描述
如果无法将所有棋子变成一个颜色,输出 "Impossible"(不含引号),否则输出变成一个颜色的最小操作次数。
示例1
输入
4 4
1001
1101
1001
1000
输出
4
题解
思路
知识点:状压枚举+递推。
首先注意到可以通过对第 \(i\) 行操作第 \(i-1\) 行全部按灭或按亮,于是可以通过这个操作将状态变换成最后一行的亮灭,而前三行统一亮灭。而基于这些操作的最后一行如果不是全灭或全亮就说明初状态不可行,否则可行。
然后,对于第一行会有 \(2^m\) 种初状态,只要遍历这些状态,分别执行上述操作即可。
具体操作上,可以用一个 \(m\) 位二进制 \(change[i]\) 表示对 \(i\) 行灯的开关操作,\(1\) 表示按,\(0\) 表示不按。对于 \(i\) 行的状态 \(a[i]\) 而言,需要将 \(1\) 的位置取反且周围 \(4\) 个也取反,\(0\) 的位置不变,即\(a[i] \oplus change[i] \oplus (change[i]>>1) \oplus (change[i]<<1)\) ;对于 \(i-1\) 行的状态 \(a[i-1]\) 而言,即 \(a[i-1] \oplus change[i]\) ,观察这个性质,我们可以将下一次按钮方式设置为本行的灯状态,做到用下一行按本行的灯;对于 \(i+1\) 行的状态 \(a[i+1]\) 而言,即 \(a[i+1] \oplus change[i]\)。需要注意 \(change[i] << 1\) 可能在int里面使范围外的二进制码改变需要加上限制 \((change[i] << 1) \& ((1<<m)-1))\) 。
最后计算 \(change[0\cdots m-1]\) 可以消去二进制最后一位 \(1\) 来计数。
时间复杂度 \(O(nm)\)
空间复杂度 \(O(nm)\)
本题为POJ1753的变种。
代码
#include <bits/stdc++.h>
using namespace std;
int n,m;
int a[107],b[107],cur[107],change[107];
int cal(int num){
int ans = 0;
while(num){
ans++;
num &= num-1;
}
return ans;
}
int sol(int a[]){
int ans = ~(1<<31);
for(change[0] = 0;change[0]<(1<<m);change[0]++){
int sum = 0;
cur[0] = a[0];
for(int i = 0;i<n;i++){
sum += cal(change[i]);
cur[i] = cur[i] ^ change[i] ^ (change[i]>>1) ^ ((change[i]<<1)&((1<<m)-1));
cur[i+1] = a[i+1] ^ change[i];
change[i+1] = cur[i];
}
if(!cur[n-1]) ans = min(ans,sum);
}
return ans;
}
int main(){
std::ios::sync_with_stdio(0);
cin>>n>>m;
for(int i = 0;i<n;i++){
for(int j = 0;j<m;j++){
char tmp;
cin>>tmp;
if(tmp == '1') a[i] |= (1<<j);
else if(tmp == '0') b[i] |= (1<<j);
}
}
int ans = min(sol(a),sol(b));
if(ans>n*m) cout<<"Impossible"<<'\n';
else cout<<ans<<'\n';
return 0;
}
NC235250 牛可乐的翻转游戏的更多相关文章
- [Swift]LeetCode293. 翻转游戏 $ Flip Game
You are playing the following Flip Game with your friend: Given a string that contains only these tw ...
- Luogu 1764 翻转游戏 - 枚举 + 搜索
题目描述 kkke在一个n*n的棋盘上进行一个翻转游戏.棋盘的每个格子上都放有一个棋子,每个棋子有2个面,一面是黑色的,另一面是白色的.初始的时候,棋盘上的棋子有的黑色向上,有的白色向上.现在kkke ...
- 牛客网 牛可乐发红包脱单ACM赛 C题 区区区间间间
[题解] 我想到了两种解法. 一种是扫描线+线段树+单调栈. 另一种方法是O(n)的,比较巧妙. 考虑每个数在哪些区间可以作为最小数.最长的区间就是它向左右走,直到有数字比它小,这个可以用单调栈维护. ...
- 【2018寒假集训 Day1】【位运算】翻转游戏
翻转游戏(flip) [问题描述] 翻转游戏是在一个 4 格×4 格的长方形上进行的,在长方形的 16 个格上每 个格子都放着一个双面的物件.每个物件的两个面,一面是白色,另一面是黑色, 每个物件要么 ...
- [LeetCode] 293. Flip Game 翻转游戏
You are playing the following Flip Game with your friend: Given a string that contains only these tw ...
- [LeetCode] 294. Flip Game II 翻转游戏 II
You are playing the following Flip Game with your friend: Given a string that contains only these tw ...
- 294. 翻转游戏 II
题目: 链接:https://leetcode-cn.com/problems/flip-game-ii/ 你和朋友玩一个叫做「翻转游戏」的游戏,游戏规则:给定一个只有 + 和 - 的字符串.你和朋友 ...
- [LeetCode] Flip Game 翻转游戏
You are playing the following Flip Game with your friend: Given a string that contains only these tw ...
- 牛客1024B 石头游戏
题目描述 石头游戏在一个 \(n\) 行 \(m\) 列 \((1\leq n,m \leq 8)(1≤n,m≤8)\) 的网格上进行,每个格子对应一种操作序列,操作序列至多有10种,分别用0~9这1 ...
随机推荐
- mouseenter 和 mouseover 的区别
当鼠标移动到元素上时就会触发mouseenter事件 类似mouseover,它们两者之间的差别是 mouseover鼠标经过自身盒子会触发,经过子盒子还会触发.mouseenter只会经过自身盒子触 ...
- 技术管理进阶——一线Leader怎么做?经理的速成宝典
原创不易,求分享.求一键三连 本期培训材料关注公众号后回复:经理培训,获得 前段时间有个同学问我有没有一线Leader的速成培训课程,很好的问题,首先我们需要定义一下什么是小Leader: 所谓小Le ...
- arduino 天下第一(暴论) -- 智能猫眼与 SDDC 连接器移植到 arduino 上
前言 之前看了官方玩过一个智能猫眼摄像头,我很有兴趣,但是那个 IDF 平台属实难整,我光安装都整了一天,网不好下载的包可能有问题.然后命令行操作也比较麻烦,我就想到了无敌的 arduino ,ESP ...
- linux 下通过fork实现后台运行进程
1 # 通常建议使用双fork方法.在每个fork处,父级退出,子级继续 2 3 #!/usr/bin/env python 4 5 import time,platform 6 7 import o ...
- 引入『客户端缓存』,Redis6算是把缓存玩明白了…
原创:微信公众号 码农参上,欢迎分享,转载请保留出处. 哈喽大家好啊,我是没更新就是在家忙着带娃的Hydra. 在前面介绍两级缓存的文章中,我们总共给出了4种实现方案,在项目中整合了本地缓存Caffe ...
- Docker系列教程01-使用Docker镜像
docker系列导读 一文带你读懂什么是docker Docker安装部署 10张图带你深入理解Docker容器和镜像 前言 学习Docker,我们需要掌握它的三大核心概念:镜像.容器和仓库. 今天先 ...
- vue3 vite 系统标题 系统名称统一配置
想要统一配置系统名称 或者其他的,需要在vue3中使用 vite 的环境变量 vite 的环境变量 需要创建两个文件(和 vite.config.js 文件同一目录) .env.development ...
- 好客租房53-context的使用
app组件要传递给child组件 该如何处理 更好的姿势 跨组件传递数据 Provider 用来提供数据 Consumer用来消费数据 1调用React.createContext() 创建provi ...
- 附001.Python多版本环境管理
一 环境背景 由于Python的版本过多,且不同版本之间差异性较大.同时又因系统底层需要调用当前版本Python,所以不能随意变更当前系统Python版本.因此,在多版本共存的情况下,Python多环 ...
- MySQL之事务和redo日志
事务 事务的四个ACID特性. Atomicity 原子性 Consistency 一致性 Isolation 隔离性 Durability 持久性 原子性 原子性即这个事务的任务要么全做了,要么全部 ...