[codevs1050]棋盘染色 2
[codevs1050]棋盘染色 2
试题描述
有一个5*N的棋盘,棋盘中的一些格子已经被染成了黑色,你的任务是对最少的格子染色,使得所有的黑色能连成一块。
输入
第一行一个整数N(<=100),接下来N行每行一个长度为5的01串,1表示所在格子已经被染成了黑色,0表示所在格子没有被染色。
输出
输出最少需要对多少个格子进行染色
输入示例
输出示例
数据规模及约定
N(<=100)
题解
状压 dp 一下。设 f(i, S) 表示考虑前 i 行,最后一行情况为集合 S 的最小代价;其中 S 是一个 5 位的 4 进制数,若某一位为 0,则表示第 i 行对应的位置没有染色,若某一位为 x(0 < x < 4),则表示该位上有染色并且该位置所在的连通分量编号为 x(注意这里的连通分量是指考虑前 i 行时的连通性,因为最终要做到整个图被弄成一个连通分量,所以任意时刻同一连通块必定能够贯穿上下),一行中只有 5 个位置,所以最多产生 3 个连通块,也就是每一位上只可能是 0, 1, 2, 3,所以是 5 位的 4 进制数。
转移时枚举一下下一行在哪些位置染色,然后暴力搞一搞判一判就好了。
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cmath>
#include <stack>
#include <vector>
#include <queue>
#include <cstring>
#include <string>
#include <map>
#include <set>
using namespace std; int read() {
int x = 0, f = 1; char c = getchar();
while(!isdigit(c)){ if(c == '-') f = -1; c = getchar(); }
while(isdigit(c)){ x = x * 10 + c - '0'; c = getchar(); }
return x * f;
} #define maxn 110
#define maxm 10
char Map[maxn][maxm];
int n, A[maxn][maxm], f[maxn][1030], now[maxm], lst[maxm], fin[maxm], tfin[maxm]; bool iszero(char* S) {
int l = strlen(S);
for(int i = 0; i < l; i++) if(S[i] - '0') return 0;
return 1;
}
int bitcal(int S) {
int cnt = 0;
while(S) cnt += (S & 1), S >>= 1;
return cnt;
}
void Up(int& a, int b) {
if(a < 0) a = b;
else a = min(a, b);
return ;
} void show(int T[]) {
for(int i = 0; i < 5; i++) printf("%d%c", T[i], i < 4 ? ' ' : '\n');
return ;
} int main() {
n = read();
for(int i = 1; i <= n; i++) scanf("%s", Map[i]); int up = 1, down = n, cnt = 0;
while(iszero(Map[up])) up++;
while(iszero(Map[down])) down--;
for(int i = up; i <= down; i++) {
cnt++;
for(int j = 0; j < 5; j++) A[cnt][j] = Map[i][j] - '0';
}
memset(f, -1, sizeof(f));
f[0][0] = 0;
int all = (1 << 5) - 1, All = (1 << 10) - 1;
for(int i = 1; i <= cnt; i++) {
int reaS = 0;
for(int j = 4; j >= 0; j--) reaS = reaS << 1 | A[i][j];
for(int S = 0; S <= all; S++) if((reaS | S) == S) {
int cnt1 = bitcal(reaS ^ S);
for(int j = 0; j < 5; j++) now[j] = S >> j & 1;
for(int tS = 0; tS <= All; tS++) if(f[i-1][tS] >= 0) {
int tmpS = tS;
for(int j = 0; j < 5; j++) lst[j] = tmpS % 4, tmpS >>= 2;
int mx = 0;
for(int j = 0; j < 5; j++) if(now[j]) {
if(!j) mx = fin[j] = 1;
else if(!fin[j-1]) fin[j] = ++mx;
else fin[j] = fin[j-1];
}
else fin[j] = 0;
// printf("%d _fin: ", cnt1); show(fin);
mx = 0;
for(int j = 0; j < 5; j++) mx = max(mx, lst[j]);
int has[maxm]; memset(has, 0, sizeof(has));
for(int j = 0; j < 5; j++) if(lst[j]) {
if(!has[lst[j]] && fin[j]) has[lst[j]] = fin[j];
else if(fin[j]) {
fin[j] = has[lst[j]];
for(int x = j - 1; x >= 0 && fin[x]; x--) fin[x] = has[lst[j]];
for(int x = j + 1; x < 5 && fin[x]; x++) fin[x] = has[lst[j]];
}
}
bool ok = 1;
for(int j = 1; j <= mx; j++) if(!has[j]) {
ok = 0; break;
}
if(!ok) continue;
for(int j = 0; j < 5; j++) tfin[j] = fin[j];
sort(tfin, tfin + 5);
int x = unique(tfin, tfin + 5) - tfin;
if(x > 1) for(int j = 0; j < 5; j++) fin[j] = lower_bound(tfin, tfin + x, fin[j]) - tfin;
// show(now); show(lst); show(fin);
tmpS = 0;
for(int j = 4; j >= 0; j--) tmpS = tmpS << 2 | fin[j];
Up(f[i][tmpS], f[i-1][tS] + cnt1);
// printf("%d\n", f[i][tmpS]); putchar('\n');
}
}
} int ans = -1;
for(int S = 0; S <= all; S++) {
for(int i = 0; i < 5; i++) now[i] = S >> i & 1;
int reaS = 0;
for(int i = 4; i >= 0; i--) reaS = reaS << 2 | now[i];
if(f[cnt][reaS] >= 0) Up(ans, f[cnt][reaS]);
}
printf("%d\n", ans); return 0;
}
写个状压 dp 真费劲。。。
[codevs1050]棋盘染色 2的更多相关文章
- [CodeVs1050]棋盘染色2(状态压缩DP)
题目大意:有一个5*N(≤100)的棋盘,棋盘中的一些格子已经被染成了黑色,求最少对多少格子染色,所有的黑色能连成一块. 这题卡了我1h,写了2.6k的代码,清明作业一坨还没做啊...之前一直以为这题 ...
- codevs——1049 棋盘染色
1049 棋盘染色 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 黄金 Gold 题解 查看运行结果 题目描述 Description 有一个5×5的棋盘,上面有一 ...
- 【wikioi】1049 棋盘染色(迭代深搜)
http://www.wikioi.com/problem/1049/ 这题我之前写没想到迭代加深,看了题解,然后学习了这种搜索(之前我写的某题也用过,,但是不懂专业名词 囧.) 迭代加深搜索就是限制 ...
- codevs 1049 棋盘染色
题目描述 Description 有一个5×5的棋盘,上面有一些格子被染成了黑色,其他的格子都是白色,你的任务的对棋盘一些格子进行染色,使得所有的黑色格子能连成一块,并且你染色的格子数目要最少.读入一 ...
- 1050 棋盘染色 2 - Wikioi
题目描述 Description 有一个5*N的棋盘,棋盘中的一些格子已经被染成了黑色,你的任务是对最少的格子染色,使得所有的黑色能连成一块. 输入描述 Input Description 第一行一个 ...
- HDU 5402 Travelling Salesman Problem(棋盘染色 构造 多校啊)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5402 Problem Description Teacher Mai is in a maze wit ...
- CODEVS——T 1049 棋盘染色
http://codevs.cn/problem/1049/ 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 黄金 Gold 题解 查看运行结果 题目描述 Descr ...
- hdu5601-N*M bulbs(黑白棋盘染色)
一个矩形,一个人从左上角走到右下角,每走过一个位置把0变成1,1变成0. 求有没有可能他离开之后所有的数都是0 假设这个矩形是一个棋盘,黑白相间. 这样会发现从一个颜色走到相同颜色可以对棋盘不产生任何 ...
- [codevs1049]棋盘染色<迭代深搜>
题目链接:http://codevs.cn/problem/1049/ 昨天的测试题里没有打出那可爱的迭代深搜,所以今天就来练一练. 这道题其实我看着有点懵,拿着题我就这状态↓ 然后我偷偷瞄了一眼hz ...
随机推荐
- Macbook air 上打开cocoscreator出错
Error: EROFS: read-only file system, open '/Volumes/Cocos Creator/CocosCreator.app/Contents/Resource ...
- ceph脚本-自动部署计算机节点
依然还在加班中,最近确实忙的脚打后脑勺! 又花了些时间丰富ceph脚本,可以连带着自动部署计算机节点了. 这一部分内容是后加的.可以关注我的公众号获取更多的项目代码和讲解!波神与你同行哦,加油!!!
- array_keys
<?php$array = array(0 => 100, "color" => "red");print_r(array_keys($arr ...
- 安装nginx的一些注意事项
1.如何彻底屏蔽掉Nginx的banner 为了安全或者某些个人的原因,如果要屏蔽掉nginx的banner,要修改以下几个位置: src/http/ngx_http_header_filter_mo ...
- 深入Docker 存储驱动 (转)
参考: http://static.dockerone.com/ppt/filedriver.html#28
- https://www.runoob.com/python/python-variable-types.html
https://www.runoob.com/python/python-variable-types.html
- Python基础4 迭代器,生成器,装饰器,Json和pickle 数据序列化
本节内容 迭代器&生成器 装饰器 Json & pickle 数据序列化 软件目录结构规范 作业:ATM项目开发 1.列表生成式,迭代器&生成器 列表生成式 孩子,我现在有个需 ...
- Qt_为什么学习Qt
1)学习GUI编程,市场上任何一款产品几乎都带有图形界面,市场上很火的Androoid.IOS编程无非也是GUI app编程,GUI编程都是差不多的,学习Qt后再学习ANdroid IOS ,那都是S ...
- h lib dll文件相关部分
参考:https://www.cnblogs.com/azbane/p/7364060.html 只对其中自己用得到的重点做了个笔记. 1..h头文件是编译时必须的,lib是链接时需要的,dll是运行 ...
- 详解Spring面向切面编程(AOP)三种实现
一.什么是AOP AOP(Aspect Oriented Programming),即面向切面编程,可以说是OOP(Object Oriented Programming,面向对象编程)的补充和完善. ...