UVA-11882 bfs + dfs + 剪枝
假设当前已经到达(x,y),用bfs判断一下还可以到达的点有maxd个,如果maxd加上当前已经经过的长度小于当前答案的长度就退出,如果相同,就将bfs搜索到的点从大到小排序,如果连最大序列都无法大于当前答案,就直接退出,否则继续搜索。
但是这题时间要求极高,不可以同时搜索同时更新答案,必须直到完成一种搜索(到达边界)才能更新答案,否则必定超时。
AC代码:
#include<cstdio> #include<queue> #include<cstring> #include<algorithm> using namespace std; const int maxn = 30 + 5; char G[maxn][maxn]; int vis[maxn][maxn], d[maxn][maxn]; int n, m; int ans[maxn], len; //答案以及答案的长度 int a[maxn]; struct node{ int x, y; node(){ } node(int x, int y):x(x), y(y){ } }; const int dx[] = {1,-1,0,0}; const int dy[] = {0,0,1,-1}; int bfs(int x, int y, int cur){ //从cur开始保存 int c = cur; memcpy(d, vis, sizeof(vis)); queue<node>q; q.push(node(x, y)); d[x][y] = 1; while(!q.empty()){ node pos = q.front(); q.pop(); x = pos.x, y = pos.y; for(int i = 0; i < 4; ++i){ int px = x + dx[i], py = y + dy[i]; if(px < 0 || px >= n || py < 0 || py >= m) continue; if(G[px][py] == '#' || d[px][py]) continue; d[px][py] = 1; a[c++] = G[px][py] - '0'; q.push(node(px, py)); } } return c; } bool cmp(int a, int b){ return a > b; } void dfs(int x, int y, int cur){ int h = bfs(x, y, cur); if(h == cur) { //边界 int flag = 0; if(cur < len) return; else if(cur > len) flag = 1; else for(int i = 0; i < len; ++i){ if(a[i] < ans[i]) return; else if(a[i] > ans[i]) { flag = 1; break; } } if(flag) { len = cur; for(int i = 0; i < len; ++i) ans[i] = a[i]; //update } return; } if(h < len) return; else if(h == len){ sort(a + cur, a + h, cmp); int flag = 0; for(int i = 0; i < len; ++i) { if(a[i] < ans[i]) return; else if(a[i] > ans[i]) { flag = 1; break; } } if(!flag) return; } for(int i = 0; i < 4; ++i){ int px = x + dx[i], py = y + dy[i]; if(px < 0 || px >= n || py < 0 || py >= m) continue; if(G[px][py] == '#' || vis[px][py]) continue; a[cur] = G[px][py] - '0'; vis[px][py] = 1; dfs(px, py, cur + 1); vis[px][py] = 0; } } int main(){ while(scanf("%d%d", &n, &m) == 2 && n){ memset(ans, 0, sizeof(ans)); len = 0; for(int i = 0; i < n; ++i) scanf("%s", G[i]); for(int i = 0; i < n; ++i) for(int j = 0; j < m; ++j) { if(G[i][j] == '#') continue; memset(vis, 0, sizeof(vis)); a[0] = G[i][j] - '0'; vis[i][j] = 1; dfs(i, j, 1); vis[i][j] = 0; } for(int i = 0; i < len; ++i){ printf("%d", ans[i]); } printf("\n"); } return 0; }
还有一种dfs代码,思路同上
#include<cstdio> #include<queue> #include<cstring> #include<algorithm> using namespace std; const int maxn = 30 + 5; char G[maxn][maxn]; int vis[maxn][maxn], d[maxn][maxn]; int n, m; int ans[maxn], len; //答案以及答案的长度 int a[maxn]; struct node{ int x, y; node(){ } node(int x, int y):x(x), y(y){ } }; const int dx[] = {1,-1,0,0}; const int dy[] = {0,0,1,-1}; bool cmp(int a, int b){ return a > b; } int bfs(int x, int y, int cur){ int c = cur; memcpy(d, vis, sizeof(vis)); queue<node>q; q.push(node(x, y)); d[x][y] = 1; while(!q.empty()){ node pos = q.front(); q.pop(); x = pos.x, y = pos.y; for(int i = 0; i < 4; ++i){ int px = x + dx[i], py = y + dy[i]; if(px < 0 || px >= n || py < 0 || py >= m) continue; if(G[px][py] == '#' || d[px][py]) continue; d[px][py] = 1; a[c++] = G[px][py] - '0'; q.push(node(px, py)); } } return c; } void dfs(int x, int y, int cur){ a[cur++] = G[x][y] - '0'; int h = bfs(x, y, cur); if(h == cur) { //边界 int flag = 0; if(cur < len) return; else if(cur > len) flag = 1; else for(int i = 0; i < len; ++i){ if(a[i] < ans[i]) return; else if(a[i] > ans[i]) { flag = 1; break; } } if(flag) { len = cur; for(int i = 0; i < len; ++i) ans[i] = a[i]; //update } return; } if(h < len) return; else if(h == len){ sort(a + cur, a + h, cmp); int flag = 0; for(int i = 0; i < len; ++i) { if(a[i] < ans[i]) return; else if(a[i] > ans[i]) { flag = 1; break; } } if(!flag) return; } vis[x][y] = 1; for(int i = 0; i < 4; ++i){ int px = x + dx[i], py = y + dy[i]; if(px < 0 || px >= n || py < 0 || py >= m) continue; if(G[px][py] == '#' || vis[px][py]) continue; dfs(px, py, cur); } vis[x][y] = 0; } int main(){ while(scanf("%d%d", &n, &m) == 2 && n){ memset(ans, 0, sizeof(ans)); len = 0; for(int i = 0; i < n; ++i) scanf("%s", G[i]); for(int i = 0; i < n; ++i) for(int j = 0; j < m; ++j) { if(G[i][j] == '#') continue; memset(vis, 0, sizeof(vis)); dfs(i, j, 0); } for(int i = 0; i < len; ++i){ printf("%d", ans[i]); } printf("\n"); } return 0; }
如有不当之处欢迎指出!
UVA-11882 bfs + dfs + 剪枝的更多相关文章
- Sticks(UVA - 307)【DFS+剪枝】
Sticks(UVA - 307) 题目链接 算法 DFS+剪枝 1.这道题题意就是说原本有一些等长的木棍,后来把它们切割,切割成一个个最长为50单位长度的小木棍,现在想让你把它们组合成一个个等长的大 ...
- hdu 1044(bfs+dfs+剪枝)
Collect More Jewels Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Othe ...
- POJ2688状态压缩(可以+DFS剪枝)
题意: 给你一个n*m的格子,然后给你一个起点,让你遍历所有的垃圾,就是终点不唯一,问你最小路径是多少? 思路: 水题,方法比较多,最省事的就是直接就一个BFS状态压缩暴搜就行 ...
- soj1091 指环王 bfs+hash+剪枝
原题链接http://acm.scu.edu.cn/soj/problem.action?id=1091 这题的主要解法就是搜索,我用的是bfs,用map将二维数组处理成字符串作为主键,到达当前状态的 ...
- Addition Chains POJ - 2248 (bfs / dfs / 迭代加深)
An addition chain for n is an integer sequence <a0, a1,a2,...,am=""> with the follow ...
- *HDU1455 DFS剪枝
Sticks Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Subm ...
- POJ 3009 DFS+剪枝
POJ3009 DFS+剪枝 原题: Curling 2.0 Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 16280 Acce ...
- poj 1724:ROADS(DFS + 剪枝)
ROADS Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 10777 Accepted: 3961 Descriptio ...
- DFS(剪枝) POJ 1011 Sticks
题目传送门 /* 题意:若干小木棍,是由多条相同长度的长木棍分割而成,问最小的原来长木棍的长度: DFS剪枝:剪枝搜索的好题!TLE好几次,终于剪枝完全! 剪枝主要在4和5:4 相同长度的木棍不再搜索 ...
随机推荐
- j2e应用相关技术
j2e应用相关技术 轻量级j2e应用以传统的jsp作为变现层技术,以一系列开源框架作为MVC层,中间件,持久层解决方案,并将这些开源框架有机组合在一起,使得j2e具有高度的可扩展性,可维护性. ser ...
- GDB 的使用
gdb使用: 1.编译时必须加-g选项,生成调试需要的信息.如 g++ xxx.cpp -o xxx -g 2.调试最好结合core文件 3.调试命令:gdb xxx x ...
- java 网络编程之UDP通信和简单的群聊程序
*/ .hljs { display: block; overflow-x: auto; padding: 0.5em; color: #333; background: #f8f8f8; } .hl ...
- Not Found The requested URL / was not found on this server.
http://www.wanysys.cc/coding/php/800.html 今天在做本地PHP项目的时候,想把之前wampserver的本地虚拟服务器环境访问方式改为本地localhost访问 ...
- 学会配置nginx
一.作为一名开发人员,大家可能经常会用到服务器,但是一般线上的服务器可能都是公司公用的,而且线上的服务器一般也不是能随随便便给个人用的,所以部署本地服务器看来是一遍必不可少的事情和能力呀,所以,ngi ...
- 【转】EI收录的中国期刊
ISSN 期刊名 0567-7718 Acta Mechanica Sinica 1006-7191 Acta Metallurgica Sinica (English Letters) ...
- shell参数传递
应用实例: #!/bin/bash #运行:bash para_tran.bash text1.txt text2.txt #"set $1"设置存储传入的第一参数 #" ...
- python --- 基础多线程编程
在python中进行多线程编程之前必须了解的问题: 1. 什么是线程? 答:线程是程序中一个单一的顺序控制流程.进程内一个相对独立的.可调度的执行单元,是系统独立调度和分派CPU的基本单位指运行中的程 ...
- sqlserver数据库使用技巧(一)--限制数据库的大小
如何限制数据库的大小? 随着数据库的使用,他占用的空间会越来越大,为了便于资源的合理分配和管理,我们可以限制其最大的大小,这个建议只在测试环境使用 具体操作如下: 打开sqlserver数据库管理工具 ...
- C#中的out参数/ref参数/params可变参数
out参数: out关键字 通过引用来传递参数,在定义方法和调用方法的时候都必须使用out关键字 简单来讲out可以用来返回多个参数类型. static void Main(string[] args ...