POJ 1984 Navigation Nightmare 二维带权并查集

2024-06-15 11:38

本文主要是介绍POJ 1984 Navigation Nightmare 二维带权并查集,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目来源:POJ 1984 Navigation Nightmare

题意:给你一颗树 k次询问 求2点之间的曼哈顿距离 并且要在只有开始k条边的情况下

思路:按照方向 我是以左上角为根 左上角为原点 dx[i]为i点距离根的x坐标 dy[]是y坐标 这两个可以通过路径压缩求出 只不过是二维而已

#include <cstdio>
#include <cstdlib>
#include <cstring>
using namespace std;
const int maxn = 40010;
int f[maxn], dx[maxn], dy[maxn];struct node
{int u, v, w;char s[3];
}a[maxn];
void init(int n)
{for(int i = 1; i <= n; i++)f[i] = i;memset(dx, 0, sizeof(dx));memset(dy, 0, sizeof(dy));
}int find(int x)
{if(x != f[x]){int rt = find(f[x]);dx[x] += dx[f[x]];dy[x] += dy[f[x]];f[x] = rt;return rt;}return x;
}
int main()
{int n, m;scanf("%d %d", &n, &m);init(n);for(int i = 1; i <= m; i++){scanf("%d %d %d %s", &a[i].u, &a[i].v, &a[i].w, a[i].s);}int q;scanf("%d", &q);int i = 1;while(q--){int s, e, w;scanf("%d %d %d", &s, &e, &w);while(i <= m && i <= w){int u = a[i].u;int v = a[i].v;int x = find(u);int y = find(v);if(x == y)continue;if(a[i].s[0] == 'E'){f[y] = x;dx[y] = dx[u]-dx[v]+a[i].w;dy[y] = dy[u]-dy[v];}else if(a[i].s[0] == 'W'){f[x] = y;dx[x] = dx[v]-dx[u]+a[i].w;dy[x] = dy[v]-dy[u];}else if(a[i].s[0] == 'N'){f[x] = y;dy[x] = dy[v]-dy[u]+a[i].w;dx[x] = dx[v]-dx[u];}else{f[y] = x;dy[y] = dy[u]-dy[v]+a[i].w;dx[y] = dx[u]-dx[v];}i++;}	int x = find(s);int y = find(e);if(x == y){int d = abs(dx[s]-dx[e]) + abs(dy[s]-dy[e]);printf("%d\n", d);}else{puts("-1");}}return 0;
}


 

这篇关于POJ 1984 Navigation Nightmare 二维带权并查集的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1063348

相关文章

随想录 Day 69 并查集 107. 寻找存在的路径

随想录 Day 69 并查集 107. 寻找存在的路径 理论基础 int n = 1005; // n根据题目中节点数量而定,一般比节点数量大一点就好vector<int> father = vector<int> (n, 0); // C++里的一种数组结构// 并查集初始化void init() {for (int i = 0; i < n; ++i) {father[i] = i;}

poj 3882(Stammering Aliens) 后缀数组 或者 hash

后缀数组:  构建后缀数组,注意要在字符串莫末尾加上一个没出现过的字符。然后可以2分或者直接扫描,直接扫描需要用单调队列来维护 VIEW CODE #include<cstdio>#include<algorithm>#include<iostream>#include<cmath>#include<queue>#include<stack>#include<string

poj 3294(Life Forms) 2分+ 后缀数组

我曾用字符串hash写,但是超时了。只能用后最数组了。大致思路:用不同的符号吧字符串连接起来,构建后缀数组,然后2分答案,依次扫描后缀数组,看是否瞒住条件。 VIEW CODE #include<cstdio>#include<vector>#include<cmath>#include<algorithm>#include<cstring>#include<cassert>#

poj 2391 Ombrophobic Bovines (网络流)

这是一道很经典的网络流的题目。首先我们考虑假如我们的时间为无穷大。我们吧每个点拆成2个点 i和i' .。虚拟源点s和汇点t。对于每个点建边(s,i, a[i])  (i‘,t,ib[i]) 。 其中a[i]为给点有多少牛,b[i]为容量。i和j连通 建边 (i,j',inf);如果最大流==所有牛的个数,就可能装下所有的牛。那么现在我们考虑时间。假设最大时间为T.那么如果i到j的的最短时间>T

poj 1330 LCA 最近公共祖先

水题目。直接上代码了。 VIEW CODE #include<cstdio>#include<algorithm>#include<iostream>#include<cmath>#include<queue>#include<stack>#include<string>#include<cstring>#include<map>#include<vector>#

poj 3160 Father Christmas flymouse 强连通+dp

首先我们可以确定的是,对于val值小于0的节点都变成0.   假设一个集合内2个房间都能任意到达,那么我就可以吧集合内的所有点的价值都取到,并且可以达到任一点。实际上集合内的每个点是相同的,这样的集合就是一个强连通分量。 那么我们就可以用tarjin算法进行强连通缩点, 最后形成一个dag的图。在dag的图上面进行dp。可以先用拓扑排序后dp。或者建反响边记忆化搜索 。 VIEW

A bug‘s life 虫子的生活(带权并查集)

题目链接: 2492 -- A Bug's Life (poj.org) 题目描述: 思路: 带权并查集,处理方法基本与食物链(http://t.csdnimg.cn/fSnRr)相同,没什么思维创新 但是一开始WA了几次,有些细节没有注意好,还是需要静下心来,好好分析问题,需要警惕! 参考代码: //#include<bits/stdc++.h>#include<iostream

05-5.5.3 并查集的进一步优化

👋 Hi, I’m @Beast Cheng 👀 I’m interested in photography, hiking, landscape… 🌱 I’m currently learning python, javascript, kotlin… 📫 How to reach me --> 458290771@qq.com 喜欢《数据结构》部分笔记的小伙伴可以订阅专栏,今后还会

Java中的二维数组和c语言中的二维数组的区别

我觉得,JAVA的多维数组其实是数组包数组,即他们下一个数组是独立的,可以独立分配内存大小,跟C语言的数组不一样,C语言的数组无论维数是多少,他们每一维的内存大小都一样。 打个比方: JAVA的三维数组 某公司有m个工厂,这个是第一维; 每个工厂有n个仓库,这个是第二维; 每个仓库有o件库存,这是第三维; 每个工厂的仓库数量都不同,每个仓库的库存数量又都不同。 通过

poj 1564 Sum It Up -- DFS 递归

题意:给一个数 t ,以及 n 个数,求 n 个数中的几个数加起来的和为 t 的情况有多少种。 注意:题目要求相同的组合方式不能出现2次,即 “3 4 1 1 1 1 ” 的结果为:“1+1+1”。 思路:一个  for  循环遍历一遍,每个 i 表示以当前数为起点开始一次DFS递归,当所遍历的和为 t 时,输出该组合方式,如果大于 t 则返回,小于则往下递归。以二维数组保存已经输出过的数据,