题目链接: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. GIS可视化——热点图

    一.简介 SuperMap iClient for JavaScript提供了热点图(HeatMapLayer),用于渲染数据衰减趋势.颜色渐变的效果. 原理:在客户端直接渲染的栅格图,热点图的渲染需 ...

  2. 【前端阅读】——《活用PHP、MySQL建构Web世界》摘记之设计技巧

    二.设计技巧 Programming的习惯因人而异,这里只提供一些经验,可以参考. 1.利用Include模块化你的程序代码 Include函数基本上说:就像是把另一个文件(HTML或者PHP程序)读 ...

  3. Mark一下, dp状态转移方程写对,可是写代码都错,poj 1651 poj 1179

    dp题: 1.写状态转移方程; 2.考虑初始化边界,有意义的赋定值.还没计算的赋边界值: 3.怎么写代码自底向上计算最优值 今天做了几个基础dp,所有是dp方程写对可是初始化以及计算写错 先是poj ...

  4. 手机APP自动化持续集成方案

    自动化测试流程 自动化测试框架

  5. java集合类型接口和实现类个人总结

    转载:http://blog.csdn.net/qingchunbusanchang/article/details/39576749 java的集合是一个比较综合的概念,相关的知识有很多的博客里面做 ...

  6. 怎样通过Html网页调用本地安卓app

    怎样使用html网页和本地app进行传递数据呢?经过研究.发现还是有方法的,总结了一下,大致有一下几种方式 一.通过html页面打开Android本地的app 1.首先在编写一个简单的html页面 & ...

  7. 六种基本DCDC变换器拓扑结构

    1.SEPIC电路 2.

  8. 非标准USBasp下载线烧录Arduino BootLoader的参数设置

    本文仅适用于BootLoader损坏且买到国产“免驱USBasp下载线”导致Arduino IDE无法识别从而不能烧写的情况.是一种略显非主流的操作方式. 因为Arduino的IDE并不支持这种免驱的 ...

  9. HDFS源码分析心跳汇报之数据块增量汇报

    在<HDFS源码分析心跳汇报之BPServiceActor工作线程运行流程>一文中,我们详细了解了数据节点DataNode周期性发送心跳给名字节点NameNode的BPServiceAct ...

  10. OpenSUSE 13.1上安装StrongSwan

    结果: 1)iOS 7.1设备能够拨IPSec VPN到StrongSwan电脑上面来 - Connect to VPN 2)iOS 设备浏览器能够訪问StrongSwan VPN所在的内网地址服务器 ...