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 相同长度的木棍不再搜索 ...
随机推荐
- 链表法解决hash冲突
/* @链表法解决hash冲突 * 大单元数组,小单元链表 */ #pragma once #include <string> using namespace std; template& ...
- hibernate 持久化对象的三个状态
Hibernate中的对象有3种状态 瞬时对象(TransientObjects).持久化对象(PersistentObjects)和离线对象(DetachedObjects也叫做脱管对象) Tran ...
- 2018-01-05-医药行业的IT革命探讨
layout: post title: 2018-01-05-医药行业的IT革命探讨 key: 20180105 tags: IT AI 医疗 modify_date: 2018-01-05 --- ...
- Jetson TX2刷机教程(原创)
Jetson TX2刷机教程 一,硬件准备 1台host主机(linux系统,最好是ubuntu64位) 1台Jetson TX2的平台 二,软件包 JetPack(Jetson SDK) 下载地址: ...
- JavaBean,List,Map,json格式之间转化方式
public class TestBean { private String id; private String name; private String password; public Stri ...
- java配置context.xml文件
2018-02-08 23:32:23 修改context.xml文件 自从学习了servlet后,每次修改里面的内容后,想要访问都要重启服务器,这样感觉很麻烦的,所以今天就教大家一个方法,只需要 ...
- codechef Dynamic GCD [树链剖分 gcd]
Dynamic GCD 题意:一棵树,字词树链加,树链gcd 根据\(gcd(a,b)=gcd(a,a-b)\) 得到\(gcd(a_1, a_2, ..., a_i) = gcd(a_1, a_1- ...
- BZOJ 1758: [Wc2010]重建计划 [暂时放弃]
今天晚上思维比较乱,以后再写写吧#include <iostream> #include <cstdio> #include <cstring> #include ...
- JSON入门看这一篇就够了
什么是JSON JSON:JavaScript Object Notation [JavaScript 对象表示法] JSON 是存储和交换文本信息的语法.类似 XML. JSON采用完全独立于任何程 ...
- Quartz动态改变任务时间
基于quartz-2.2 的动态任务调度 Quartz是一个完全由java编写的开源作业调度框架. 调度器 Quartz框架的核心是调度器.调度器负责管理Quartz应用运行时环境.调度器不是靠自己做 ...