题目链接: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. 上传项目至svn服务器,从svn上获取项目

    1.在桌面右键->TortoiseSVn->Repo_brower->输入地址,进入 ,ok 2.在地址目录上右键==>>add folder==>>选择你要 ...

  2. vue DOM模板解析

    当使用 DOM 作为模板时 (例如,使用 el 选项来把 Vue 实例挂载到一个已有内容的元素上),你会受到 HTML 本身的一些限制,因为 Vue 只有在浏览器解析.规范化模板之后才能获取其内容.尤 ...

  3. CentOS6 图形界面(gnome)安装(转)

    CentOS6相对于CentOS5的安装有了不少的进步,有不少默认的选项可以选择,如: Desktop :基本的桌面系统,包括常用的桌面软件,如文档查看工具. Minimal Desktop :基本的 ...

  4. cmake学习之- cmake_parse_arguments

    最后更新: 2019-06-08 一.指令介绍 cmake_parse_arguments 为解析函数(function)或 宏(macros) 参数的命令: cmake_parse_argument ...

  5. erlang处理mongodb日期时间格式data类型(原)

    在项目中,mongo中要创建日期类型,根据这个日期类型进而对mongo设置过期时间expire,加上对应的index索引自动删除. 而mongo中的日期类型,使用ISO格式,例如:ISODate(&q ...

  6. [Linux] 概念

    操作系统包括: 内核:管理硬件资源 库:没有执行入口的程序,用于提升软件开发效率 应用程序:有执行入口的程序 常见库文件: windows系统:dll(dynamic link library)动态链 ...

  7. android 自定图库(转)

    githup: https://github.com/pengjianbo/GalleryFinal GalleryFinal简介 Android自定义相册,实现了拍照.图片选择(单选/多选). 裁剪 ...

  8. Oracle学习第三篇—多行函数

    0 order by asc/desc 默认升序 order by 列的名字|表达式|别名|序号 把空放在后边:order by desc nulls last 1分组函数--会自动滤空值 count ...

  9. 认识XmlReader

    认识XmlReader   摘要 XmlReader类是组成.NET的关键技术之一,极大地方便了开发人员对Xml的操作.通过本文您将对XmlReader有一个很好的认识,并将其应用到实际开发中. 目录 ...

  10. IDEA下使用Jetty进行Debug模式调试

    过程例如以下: (1)找到选项卡中的 –Run– 然后找到 –Edit Configurations (2)点击下图中绿色的plus–找到Maven点进去 (3)依照下边的方式在Command lin ...