POJ - 1984 Navigation Nightmare 种类并查集
思路:记录每个点与其根结点的横向距离和纵向距离,当知道其父节点与根结点的关系,很容易推出当前节点与根结点的关系:
直接相加即可。
int p = a[x].par; a[x].dx += a[p].dx; a[x].dy += a[p].dy;
合并什么的就不多说了,很容易得出。
值得一说的就是需要按照合并节点的顺序把询问排序,最后按照询问的顺序再排一次序。
AC代码
#include <cstdio> #include <cmath> #include <cctype> #include <algorithm> #include <cstring> #include <utility> #include <string> #include <iostream> #include <map> #include <set> #include <vector> #include <queue> #include <stack> using namespace std; #pragma comment(linker, "/STACK:1024000000,1024000000") #define eps 1e-10 #define inf 0x3f3f3f3f #define PI pair<int, int> typedef long long LL; const int maxn = 40000 + 5; struct node{ int par; int dx, dy; }a[maxn]; struct Edge{ int u, v, dis, ch; }b[maxn]; struct Question{ int u, v, t, num; }q[10000+5]; void init(int n) { for(int i = 1; i <= n; ++i) { a[i].par = i; a[i].dx = a[i].dy = 0; } } int find(int x) { if(a[x].par == x) return x; int r = find(a[x].par); int p = a[x].par; a[x].dx += a[p].dx; a[x].dy += a[p].dy; return a[x].par = r; } void unionset(int x, int y, int dx, int dy) { int rx = find(x), ry = find(y); if(rx != ry) { a[rx].par = y; a[rx].dx = dx - a[x].dx; a[rx].dy = dy - a[x].dy; } } int get_dis(int x, int y) { int rx = find(x), ry = find(y); if(rx != ry) return -1; return abs(a[x].dx - a[y].dx) + abs(a[x].dy - a[y].dy); } bool cmp1(const Question &a, const Question &b) { return a.t < b.t; } bool cmp2(const PI &a, const PI &b) { return a.second < b.second; } int main() { int n, m, Q; while(scanf("%d%d", &n, &m) == 2) { init(n); for(int i = 1; i <= m; ++i) { scanf("%d %d %d %c", &b[i].u, &b[i].v, &b[i].dis, &b[i].ch); //printf("%d %d %d %c\n", b[i].u, b[i].v, b[i].dis, b[i].ch); } scanf("%d", &Q); for(int i = 1; i <= Q; ++i) { scanf("%d%d%d", &q[i].u, &q[i].v, &q[i].t); q[i].num = i; } vector<PI>ans; sort(q+1, q+Q+1, cmp1); int ind = 1; for(int i = 1; i <= m; ++i) { int dx, dy; switch(b[i].ch) { case 'E': dx = b[i].dis, dy = 0; break; case 'W': dx = -b[i].dis, dy = 0; break; case 'N': dx = 0, dy = b[i].dis; break; case 'S': dx = 0, dy = -b[i].dis; break; } unionset(b[i].u, b[i].v, dx, dy); while(ind <= Q && q[ind].t == i) { ans.push_back(make_pair(get_dis(q[ind].u, q[ind].v), q[ind].num)); ++ind; } } sort(ans.begin(), ans.end(), cmp2); for(int i = 0; i < ans.size(); ++i) printf("%d\n", ans[i].first); } return 0; }
如有不当之处欢迎指出!
POJ - 1984 Navigation Nightmare 种类并查集的更多相关文章
- POJ1984 Navigation Nightmare —— 种类并查集
题目链接:http://poj.org/problem?id=1984 Navigation Nightmare Time Limit: 2000MS Memory Limit: 30000K T ...
- POJ 1984 Navigation Nightmare 【经典带权并查集】
任意门:http://poj.org/problem?id=1984 Navigation Nightmare Time Limit: 2000MS Memory Limit: 30000K To ...
- poj 1182:食物链(种类并查集,食物链问题)
食物链 Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 44168 Accepted: 12878 Description ...
- poj 1182 食物链(种类并查集 ‘初心者’)
题目链接:http://poj.org/problem?id=1182 借着这题可以好好理解一下种类并查集,这题比较简单但挺经典的. 题意就不解释了,中问题. 关于种类并查集结局方法也是挺多的 1扩增 ...
- POJ 1984 Navigation Nightmare 带全并查集
Navigation Nightmare Description Farmer John's pastoral neighborhood has N farms (2 <= N <= ...
- POJ 1984 - Navigation Nightmare - [带权并查集]
题目链接:http://poj.org/problem?id=1984 Time Limit: 2000MS Memory Limit: 30000K Case Time Limit: 1000MS ...
- POJ 1984 Navigation Nightmare(二维带权并查集)
题目链接:http://poj.org/problem?id=1984 题目大意:有n个点,在平面上位于坐标点上,给出m关系F1 F2 L D ,表示点F1往D方向走L距离到点F2,然后给出一系 ...
- poj 1984 Navigation Nightmare(带权并查集+小小的技巧)
题目链接:http://poj.org/problem?id=1984 题意:题目是说给你n个线,并告知其方向,然后对于后面有一些询问,每个询问有一个时间点,要求你输出在该时间点a,b的笛卡尔距离,如 ...
- POJ 1182 食物链(种类并查集)
食物链 Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 63592 Accepted: 18670 Description ...
随机推荐
- junit测试套件
在实际项目中,随着项目进度的开展,单元测试类会越来越多,可是直到现在我们还只会一个一个的单独运行测试类,这在实际项目实践中肯定是不可行的.为了解决这个问题,JUnit 提供了一种批量运行测试类的方法, ...
- hibernate_xml映射exception
错误原因:在通过hibernate指示生成两个表之间的外键关系之后,一个表中引用的外键不在另一个表的参考范围里面. 解决:使之满足参考完整性 org.hibernate.TransientObject ...
- gb_tree平衡树源码
1.平衡树简称AVL,出名的有红黑树,这里介绍一下gb_tree的实现 gb_tree的原理比红黑树简单,没有过多的旋转跳跃闭着眼,是一种叫AA树的结构(Arne Andersson's Genera ...
- 程序员之殇 —— (Are you afraid of me? Don't be.)灵感=神秘感
Are you afraid of me? (你们怕我吗?) Don't be.(不用怕) I am a programmer who just won't die.(我是不会死的程序员) 自从跟踪到 ...
- JFinal源码 分析之 Core包分析
ActionHandler.java 这个类继承了上面 说的Handler类,首先我们 上 几个属性 ,下面几个 属性我们 需要 关心哪些东西 呢?首先 是ActionMapping和RenderMa ...
- selenium模拟浏览器对搜狗微信文章进行爬取
在上一篇博客中使用redis所维护的代理池抓取微信文章,开始运行良好,之后运行时总是会报501错误,我用浏览器打开网页又能正常打开,调试了好多次都还是会出错,既然这种方法出错,那就用selenium模 ...
- vue打包后不使用服务器直接访问方法
根据官网打包执行npm run build 后dist文件夹打开的index.html 是空白 需要开启http服务器才能访问,以下是解决办法 1.找到config文件夹下的index文件 修改成 2 ...
- StringBuffer和String需要注意的
首先,StringBuffer的toString方法和String的subString方法都是在新生成了一个新的String. 最近做的一个功能,多线程的从SQLite数据库中读取数据.将数据拼成在M ...
- Linux 下编写服务器程序时关于Address already in use 的小错误
新手,,学习linux服务器编程的时候,bind()函数出现了Address already in use 的错误,这是因为上一次bind过后,还未释放,,只要在socket和bind之间加一个函数就 ...
- Android逆向之so的半自动化逆向
因为工作需要,转型干android逆向,有几个月了.不过对于so的逆向,任然停留在,难难难的阶段,虽然上次自己还是逆向了一个15k左右的小so文件,但是,那个基本是靠,一步一步跟代码,查看堆栈信息来自 ...