POJ 1753,题目链接http://poj.org/problem?id=1753

题意:

有4*4的正方形,每个格子要么是黑色,要么是白色,当把一个格子的颜色改变(黑->白或者白->黑)时,其周围上下左右(如果存在的话)的格子的颜色也被反转,问至少反转几个格子可以使4*4的正方形变为纯白或者纯黑?

思路:

1. 每一个位置只有两种颜色,翻偶数次等于没有翻,所以只有翻基数次对棋盘有影响,即只用考虑一个位置最多翻一次。

2. 一共有16个位置,所以最多翻16次。那么可能翻0次就成功、或者翻1次成功、或者翻2次成功...或者翻16次成功。

3. 每个位置翻转的顺序对结果无影响。

那么这就变成了一个组合数问题:

将输入的16个元素放到一个数组中,进行组合数计算即可。(组合数+枚举)

代码1:

/*
* Memory: 384K
* Time: 16MS
*/
#include <cstdio>
#include <cstdlib>
#define ALLNUM 16 static int count = 0;
/*
* 15 14 13 12
* 11 10 9 8
* 7 6 5 4
* 3 2 1 0
*/
void flip(int& data, int posIdx)
{
data ^= 1<<posIdx;
//up
if (posIdx < 12) data ^= 1<<(posIdx+4);
//down
if (posIdx > 3) data ^= 1<<(posIdx-4);
//left
if (posIdx%4+1 != 4) data ^= 1<<(posIdx+1);
//right
if (posIdx%4 != 0) data ^= 1<<(posIdx-1);
} bool isOK(int data)
{
if (data == 0 || data == 0xffff) return true;
else return false;
} /*
* num 1 -- ALLNUM
* starIdx 0 -- ALLNUM-1
*/
void func(int num, int starIdx, int data, int step)
{
// printf("-----------comein >> starIdx=%d, num=%d\n", starIdx, num);
if (num == 1){
++step;
for (int i=starIdx; i<ALLNUM; ++i){
int temp = data;
flip(temp, i);
if (isOK(temp)) {
printf("%d\n", step);
exit(0);
} //log
++count;
}
}
else if (starIdx+num <= ALLNUM){ // 14 + 2 <= 16 (14 15)
++step;
for (int i=starIdx; i<=ALLNUM-num; ++i){
int temp = data;
flip(temp, i);
func(num-1, i+1, temp, step);
}
}
// else {
// printf("Error Branch----------- >> starIdx=%d, num=%d\n", starIdx, num);
// }
} // black 0, white 1
int main()
{
int num = ALLNUM;
int data = 0;
char c;
while (true)
{
scanf("%c", &c);
if (c == '\n') continue;
--num;
if (c == 'w') data ^= 1<<num;
if (num == 0) break;
} if (isOK(data)) {
printf("0\n");
return 0;
} for (int i=1; i<=ALLNUM; ++i)
{
func(i, 0, data, 0);
// printf("i=%d,AllCount=%d\n", i, count);
} printf("Impossible\n");
return 0;
}

代码2:

//356K 313MS
#include <cstdio>
#include <cstdlib>
#define ALLNUM 16
bool isOK(int data)
{
if (data == 0 || data == 0xffff) return true;
else return false;
}
void flip(int& data, int posIdx)
{
data ^= 1<<posIdx;
//up
if (posIdx < 12) data ^= 1<<(posIdx+4);
//down
if (posIdx > 3) data ^= 1<<(posIdx-4);
//left
if (posIdx%4+1 != 4) data ^= 1<<(posIdx+1);
//right
if (posIdx%4 != 0) data ^= 1<<(posIdx-1);
}
// black 0, white 1
int main()
{
int num = ALLNUM;
int data = 0;
char c;
while (true)
{
scanf("%c", &c);
if (c == '\n') continue;
--num;
if (c == 'w') data ^= 1<<num;
if (num == 0) break;
}
int step = ALLNUM;
bool bImpossible = true;
for (int i=0; i<=0xffff; ++i)
{
int tempData = data;
int tempStep = 0;
for (int idx=0; idx<ALLNUM; ++idx){
if ((1<<idx) & i){ //idx位置需要翻转
flip(tempData, idx);
++tempStep;
}
}
if (isOK(tempData) && tempStep < step) step = tempStep;
}
if (step == ALLNUM)
printf("Impossible\n");
else
printf("%d\n", step);
return 0;
}

poj1753解题报告(枚举、组合数)的更多相关文章

  1. [雅礼NOIP集训 2017] number 解题报告 (组合数+二分)

    题解: 令$S(i)={i+1,i+2,...,i<<1}$,f(i,k)表示S(i)中在二进制下恰好有k个1的数的个数 那么我们有$f(i,k)=\sum_{x=1}^{min(k,p) ...

  2. HDU 4869 Turn the pokers (2014多校联合训练第一场1009) 解题报告(维护区间 + 组合数)

    Turn the pokers Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) ...

  3. 北大ACM试题分类+部分解题报告链接

    转载请注明出处:優YoU http://blog.csdn.net/lyy289065406/article/details/6642573 部分解题报告添加新内容,除了原有的"大致题意&q ...

  4. LeetCode :1.两数之和 解题报告及算法优化思路

    最近开始重拾算法,在 LeetCode上刷题.顺便也记录下解题报告以及优化思路. 题目链接:1.两数之和 题意 给定一个整数数组 nums 和一个目标值 target,请你在该数组中找出和为目标值的那 ...

  5. 20161005 NOIP 模拟赛 T2 解题报告

    beautiful 2.1 题目描述 一个长度为 n 的序列,对于每个位置 i 的数 ai 都有一个优美值,其定义是:找到序列中最 长的一段 [l, r],满足 l ≤ i ≤ r,且 [l, r] ...

  6. 【第40套模拟题】【noip2011_mayan】解题报告【map】【数论】【dfs】

    目录:1.潜伏者 [map] 2.Hankson的趣味题[数论]3.mayan游戏[dfs] 题目: 1. 潜伏者(spy.pas/c/cpp)[问题描述]R 国和S 国正陷入战火之中,双方都互派间谍 ...

  7. 2011 ACM-ICPC 成都赛区解题报告(转)

    2011 ACM-ICPC 成都赛区解题报告 首先对F题出了陈题表示万分抱歉,我们都没注意到在2009哈尔滨赛区曾出过一模一样的题.其他的话,这套题还是非常不错的,除C之外的9道题都有队伍AC,最终冠 ...

  8. 2014 ACM/ICPC 鞍山赛区现场赛 D&amp;I 解题报告

    鞍山现场赛结束了呢-- 我们出的是D+E+I三道题-- 吾辈AC掉的是D和I两道,趁着还记得.先在这里写一写我写的两道水题D&I的解题报告吧^_^. D题的意思呢是说星云内有一堆排成一条直线的 ...

  9. POJ 3126 Prime Path 解题报告(BFS & 双向BFS)

    题目大意:给定一个4位素数,一个目标4位素数.每次变换一位,保证变换后依然是素数,求变换到目标素数的最小步数. 解题报告:直接用最短路. 枚举1000-10000所有素数,如果素数A交换一位可以得到素 ...

随机推荐

  1. c++ 概念及学习/c++ concept&learning(二)

    上篇内容讲述了整个语言的发展[为什么会产生编程语言],以及学习C++所需要掌握的内容.这节开始认识第一部分最基本的内容:C++的内建类型,也就是基本类型. 在这些知识之前留一个问题:为什么基本所有语言 ...

  2. matlab的&和&&操作

    A&B(1)首先判断A的逻辑值,然后判断B的值,然后进行逻辑与的计算.(2)A和B可以为矩阵(e.g. A=[1 0],B=[0 0]).A&&B(1)首先判断A的逻辑值,如果 ...

  3. Java Client for Google Cloud Storage

    关于Google Cloud Storage Google Cloud Storage有益于大文件的存储与服务(serve).此外,Cloud Storage提供了对访问控制列表(ACLs)的使用,提 ...

  4. poj1001 Exponentiation

    Description Problems involving the computation of exact values of very large magnitude and precision ...

  5. java动态代理与老式AOP实现

    JAVA的动态代理 代理模式是常用的java设计模式,他的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息.过滤消息.把消息转发给委托类,以及事后处理消息等.代理类与委托类之间通常会 ...

  6. 【转】Nginx系列(四)--工作原理

    原博文出于:    http://blog.csdn.net/liutengteng130/article/details/46724081  感谢! 上篇文章介绍了Nginx框架的设计之管理进程以及 ...

  7. 在linux中使用find

    最近用到了在linux系统中查找文件这个任务,学习了下面两篇文章,尤其是过滤结果信息的小技巧.记录下来,省得忘了再找. http://www.360doc.com/content/10/1110/10 ...

  8. linux下编译lua

    curl -R -O http://www.lua.org/ftp/lua-5.2.3.tar.gz 编译代码时,遇到如下错误 /usr/lib/libreadline.so: undefined r ...

  9. POJ 3660 Cow Contest(传递闭包floyed算法)

    Cow Contest Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 5989   Accepted: 3234 Descr ...

  10. Educational Codeforces Round 10 D. Nested Segments (树状数组)

    题目链接:http://codeforces.com/problemset/problem/652/D 给你n个不同的区间,L或者R不会出现相同的数字,问你每一个区间包含多少个区间. 我是先把每个区间 ...