poj2965 The Pilots Brothers' refrigerator —— 技巧性
题目链接: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 —— 技巧性的更多相关文章
- POJ2965——The Pilots Brothers' refrigerator
The Pilots Brothers' refrigerator Description The game “The Pilots Brothers: following the stripy el ...
- [POJ2965]The Pilots Brothers' refrigerator (搜索/位运算)
题意 游戏“The Pilots Brothers:跟随有条纹的大象”有一个玩家需要打开冰箱的任务. 冰箱门上有16个把手.每个手柄可以处于以下两种状态之一:打开或关闭.只有当所有把手都打开时,冰箱才 ...
- poj2965 The Pilots Brothers' refrigerator
题目链接:http://poj.org/problem?id=2965 分析:1.这道题和之前做的poj1753题目差不多,常规思路也差不多,但是除了要输出最少步数外,还要输出路径.做这道题的时候在怎 ...
- POJ 2965 The Pilots Brothers' refrigerator 暴力 难度:1
The Pilots Brothers' refrigerator Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 16868 ...
- POJ 2965. The Pilots Brothers' refrigerator 枚举or爆搜or分治
The Pilots Brothers' refrigerator Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 22286 ...
- POJ2965The Pilots Brothers' refrigerator(枚举+DFS)
The Pilots Brothers' refrigerator Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 22057 ...
- The Pilots Brothers' refrigerator(dfs)
The Pilots Brothers' refrigerator Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 19718 ...
- 枚举 POJ 2965 The Pilots Brothers' refrigerator
题目地址:http://poj.org/problem?id=2965 /* 题意:4*4的矩形,改变任意点,把所有'+'变成'-',,每一次同行同列的都会反转,求最小步数,并打印方案 DFS:把'+ ...
- 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 ...
随机推荐
- 转: 为什么做java的web开发我们会使用struts2,springMVC和spring这样的框架?
from: https://github.com/RubyLouvre/agate/issues/8 今年我一直在思考web开发里的前后端分离的问题,到了现在也颇有点心得了,随着这个问题的深入,再加以 ...
- linux中expr用法
名称:expr ### 字串长度 shell>> expr length "this is a test" 14 ### 数字商数 shell>> ...
- nginx如何设置防盗链
关于nginx防盗链的方法网上有很多教程,都可以用,但是我发现很多教程并不完整,所做的防盗链并不是真正的彻底的防盗链! 一般,我们做好防盗链之后其他网站盗链的本站图片就会全部失效无法显示,但是您如果通 ...
- linux 文件查找实用技巧
1.tail catalina.out -n 100000 | grep -niR com.uujimu.utils.ArticleContentReplace.replacNumToA 查找内容,并 ...
- CNN卷积神经网络新想法
近期一直在看卷积神经网络,想改进改进弄出点新东西来.看了好多论文,写了一篇综述.对深度学习中卷积神经网络有了一些新认识,和大家分享下. 事实上卷积神经网络并非一项新兴的算法.早在上世纪八十年代就已经被 ...
- 【转载】ASP.NET之旅--深入浅出解读IIS架构
在学习Asp.net时,发现大多数作者都是站在一个比较高的层次上讲解Asp.Net. 他们耐心. 细致地告诉你如何一步步拖放控件. 设置控件属性.编写 CodeBehind代码,以实现某个特定的功能. ...
- null的比较问题
select count(*) from table_a WHERE status=1 and end_time<now(); 写这个sql的时候有点纠结,万一end_time是null怎么办 ...
- HDFS源码分析之数据块Block、副本Replica
我们知道,HDFS中的文件是由数据块Block组成的,并且为了提高容错性,每个数据块Block都会在不同数据节点DataNode上有若干副本Replica.那么,什么是Block?什么又是Replic ...
- 【puppeteer+Node.js安装环境】之WebStorm编辑器运行失败问题:Error: Cannot find module 'puppeteer'并且代码出不来“asnyc”标识以及有红色波浪线解决办法
现象一: module.js:557 throw err; ^ Error: Cannot find module 'puppeteer' at Function.Module._r ...
- NUTCH2.3 hadoop2.7.1 hbase1.0.1.1 solr5.2.1部署(二)
Precondition: hadoop 2.7.1 hbase 1.0.1.1 / hbase 0.98.13 192.168.1.106 ->master 192.168.1.105 ...