[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 ...
随机推荐
- [已读]响应式web设计实践
薄的一本,彩印,书质量和内容都不错. 响应设计三要素:媒体查询.流动布局.自适应图片.
- Object类的几个方法
1.protected Object clone()创建并返回此对象的一个副本. 2. boolean equals(Object obj)指示其他某个对象是否与此对象“相等”. 3. protect ...
- window服务 调试步骤
方法一: 1.编译windows服务项目工程 2.把服务注册到系统服务上 3.在visual studio 编辑器中,打断点,用 Debug 进程调试 方法二: 在Onstart 方法中,加上 De ...
- 多线程wait和notify实现1212
package threadT; public class ThreadMain { public static void main(String args[]) { final Object obj ...
- VC++编译出错:LNK1123: 转换到 COFF 期间失败: 文件无效或损坏
解决方法: 1.搜索C盘下的cvtres.exe,结果得到类似这样的列表: C:\Program Files\Microsoft Visual Studio 10.0\VC\bin C:\Window ...
- 自定义 TypeHandler
自定义TypeHandler分为三个步骤: 1.编写自定义TypeHandler,并继承自抽象类BaseTypeHandler<T>,实现抽象方法 2.在mybatis-config.xm ...
- Failed to obtain lock on file /usr/local/nagios/var/ndo2db.lock: Permission denied : Permission denied
Failed to obtain lock on file /usr/local/nagios/var/ndo2db.lock: Permission denied : Permission den ...
- Netbeans Makefile: recipe for target 'XXX' failed 运行failed(退出值 -1073741511 找不到C/C++库文件,关键字
今天不知怎么的又出错了 吐血了 找不到NULL new等关键字了 看到知乎上有人和我一个问题,怎么办?很简单卸载了以前的cygwin和netbeans然后重装,我重装时没有把以前的cygwin ...
- DROP SCHEMA - 删除一个模式
SYNOPSIS DROP SCHEMA name [, ...] [ CASCADE | RESTRICT ] DESCRIPTION 描述 DROP SCHEMA 从数据库中删除模式. 模式只能被 ...
- uva1614 Hell on the Markets
贪心部分的理论依据:前i个数可以凑出1-sum[i]的所有整数. 证明:第二类数学归纳,n=1时成立,假设n=k之前所有项都成立,当n=k+1时.sum[k+1]=sum[k]+a[k+1].只需证明 ...