题目链接:http://poj.org/problem?id=2965

题解:自己想到的方法是枚举搜索,结果用bfs和dfs写都超时了。网上拿别人的代码试一下只是刚好不超时的,如果自己的代码在某些方面不够优化,那超时很正常。看来这题用dfs和bfs都不是好办法。 然后又看到比较厉害的技巧:“可知翻偶数次等于没翻,即没有翻的必要,翻奇数次的结果与翻一次的结果一样“”。有了这个重要结论,那么可以具体操作了:设一个二维数组以记录每个的翻转次数。对于每个‘+’,都翻转其所在的行列(注意‘+’只翻一次),然后更新翻转次数。最后统计翻转次数为奇数的,即为实际需要翻转的。(为什么我没想到这种方法。需要仔细体会这种思维)

代码如下:

 #include<stdio.h>//poj2965
#include<string.h>
int main()
{
int a[][],sum;
char map[][];//map开一维已足够
memset(a,,sizeof(a));
for(int i = ; i<; i++)
{
scanf("%s",map[i]);
for(int j = ; j<; j++)
{
if(map[i][j]=='+')
{
a[i][j]--;//‘+’在下面的步骤中翻了两次,要减少一次
for(int k = ; k<; k++)
{
a[k][j]++;
a[i][k]++;
}
}
}
}
sum = ;
for(int i = ; i<; i++)
for(int j = ; j<; j++)
{
if(a[i][j]%) sum++;
}
printf("%d\n",sum); for(int i = ; i<; i++)
for(int j = ; j<; j++)
{
if(a[i][j]%)
printf("%d %d\n",i+,j+);
}
}

对了,附上自己wa了的dfs和bfs;

dfs:

 #include<cstdio>//poj 2965dfs 超时未过
#include<cstring>
#define MIN(a,b) (a<b?a:b) int a[],ans[]; int get()
{
for(int i = ; i<; i++)
if(a[i]) return ;
return ;
} void flip(int loc)
{
int x = loc/, y = loc%;
a[loc] = !a[loc];
for(int k = ; k<; k++)
a[x*+k] = !a[x*+k];
for(int k = ; k<; k++)
a[k*+y] = !a[k*+y];
} int dfs(int loc,int step,int tar)
{
if(loc>) return ; if(step==tar && get())
return ; else
{
flip(loc);
ans[step] = loc;
if(dfs(loc+,step+,tar)) return ;
flip(loc); if(dfs(loc+,step,tar)) return ; }
return ;
} int main()
{
int t;
char s[]; memset(ans,-,sizeof(ans));
for(int i = ; i<; i++)
{
scanf("%s",s);
for(int j = ; j<; j++)
{
if(s[j]=='+') a[i*+j] = ;
else a[i*+j] = ;
}
}
int i;
for(i = ; i<=; i++)
{
if(dfs(,,i)) break;
}
printf("%d\n",i);
for(int k = ; k<i; k++)
{
if(ans[k]!=-)
printf("%d %d\n",ans[k]/+, ans[k]%+);
} return ;
}

bfs:(bfs的队列可能不够大,溢出。用stl的话又怎么记录路径呢?)

 #include<cstdio>//poj 2965 bfs 未过
#include<cstring>
#include<queue> using namespace std; int ss,vis[],pre[];
struct node
{
int status, step,loc;
};
node q[]; void calcul( node *next,int i)
{
if(next->status&(<<i)) next->status -= (<<i);
else next->status += (<<i);
} void turn (node *next,int i)
{
calcul(next,i);
int xi = i/, yi = i%;
for(int k = ; k<; k++)
calcul(next,xi*+k);
for(int k = ; k<; k++)
calcul(next,k*+yi);
} int bfs()
{
int front = , rear = ;
q[front].status = ss, q[front].step = ;
vis[q[front].status] = ;
if(q[front].status==)
return front;
memset(vis,,sizeof(vis));
memset(pre,,sizeof(pre));
while(front<rear)
{
for(int i = ; i<; i++)
{
q[rear].status = q[front].status;
turn(&q[rear],i);
if(vis[q[rear].status]) continue; q[rear].loc = i;
q[rear].step = q[front].step + ;
pre[rear] = front;
vis[q[rear].status] = ; if(q[rear].status==) return rear;
rear++;
}
front++;
}
return -;
} void pri_path(int rear)
{
if(q[rear].step>) pri_path(pre[rear]);
printf("%d %d\n",q[rear].loc/+, q[rear].loc%+);
} int main()
{
int t;
char s[];
ss = ; for(int i = ; i<; i++)
{
scanf("%s",s);
for(int j = ; j<; j++)
{
if(s[j]=='+') ss += <<(i*+j);
}
}
int rear = bfs();
printf("%d\n",q[rear].step); pri_path(rear);
return ;
}

poj2965 The Pilots Brothers' refrigerator —— 技巧性的更多相关文章

  1. POJ2965——The Pilots Brothers' refrigerator

    The Pilots Brothers' refrigerator Description The game “The Pilots Brothers: following the stripy el ...

  2. [POJ2965]The Pilots Brothers' refrigerator (搜索/位运算)

    题意 游戏“The Pilots Brothers:跟随有条纹的大象”有一个玩家需要打开冰箱的任务. 冰箱门上有16个把手.每个手柄可以处于以下两种状态之一:打开或关闭.只有当所有把手都打开时,冰箱才 ...

  3. poj2965 The Pilots Brothers' refrigerator

    题目链接:http://poj.org/problem?id=2965 分析:1.这道题和之前做的poj1753题目差不多,常规思路也差不多,但是除了要输出最少步数外,还要输出路径.做这道题的时候在怎 ...

  4. POJ 2965 The Pilots Brothers' refrigerator 暴力 难度:1

    The Pilots Brothers' refrigerator Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 16868 ...

  5. POJ 2965. The Pilots Brothers' refrigerator 枚举or爆搜or分治

    The Pilots Brothers' refrigerator Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 22286 ...

  6. POJ2965The Pilots Brothers' refrigerator(枚举+DFS)

    The Pilots Brothers' refrigerator Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 22057 ...

  7. The Pilots Brothers' refrigerator(dfs)

    The Pilots Brothers' refrigerator Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 19718 ...

  8. 枚举 POJ 2965 The Pilots Brothers' refrigerator

    题目地址:http://poj.org/problem?id=2965 /* 题意:4*4的矩形,改变任意点,把所有'+'变成'-',,每一次同行同列的都会反转,求最小步数,并打印方案 DFS:把'+ ...

  9. The Pilots Brothers' refrigerator 分类: POJ 2015-06-15 19:34 12人阅读 评论(0) 收藏

    The Pilots Brothers' refrigerator Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 20304 ...

随机推荐

  1. win10中以管理员身份启动notepad、cmd、editplus

    win10中以管理员身份启动notepad.cmd 在开始菜单中输入,出现了之后再进行右键点击,选择管理员身份运行: 而且editplus也可以“管理员身份运行”,再也不用担心我改不了hosts了: ...

  2. haifeng

    [root@localhost 桌面]# yum list|grep wubi ibus-table-chinese-wubi-haifeng.noarch -.el7 base ibus-table ...

  3. websocket关于禁止一个账号多窗口链接的问题

    通过websocket的session.getSessionId()与oldSession.getSessionId()来equals判断是否是新窗口. 如果不同不让链接. 问题1.虽然新来的链接连不 ...

  4. mysql报错1872: Slave failed to initialize relay log info structure from the repository

    ERROR 1872 (HY000): Slave failed to initialize relay log info structure from the repository 在一台主机上增加 ...

  5. SVN 服务端、客户端安装及配置、导入导出项目

    http://blog.csdn.net/xcy13638760/article/details/12994923 http://www.cnblogs.com/armyfai/p/3985660.h ...

  6. objc_msgSend 报错

    NSMutableArray * mutableArray = [NSMutableArray arrayWithArray:array]; objc_msgSend(mutableArray,@se ...

  7. 一个简单的HTML5摇一摇实例

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xht ...

  8. Python中strip方法的妙用

    [开胃小菜] 当提到python中strip方法,想必凡接触过python的同行都知道它主要用来切除空格.有下面两种方法来实现. 方法一:用内置函数 #<python> if __name ...

  9. 搜狐新闻APP是如何使用HUAWEI DevEco IDE快速集成HUAWEI HiAI Engine

    6月12日,搜狐新闻APP最新版本在华为应用市场正式上线啦! 那么,这一版本的搜狐新闻APP有什么亮点呢? 先抛个图,来直接感受下—— ​ 模糊图片,瞬间清晰! 效果杠杠的吧. 而藏在这项神操作背后的 ...

  10. Appium python unittest pageobject如何实现加载多个case

    学习了Appium python项目施展的课程小伙伴都会有一个疑问,说现在所有的case都是通过一个suite进行一个方法一个方法进行添加的,但是在实际过程中我们不希望这样,我们做出来的功能是这样: ...