题目链接  2016 ACM-ICPC EC-Final Problem G

题意  给定一个无向图。每个点有一种颜色。

   现在给定$q$个询问,每次询问$x$和$w$,求所有能通过边权值不超过$w$的边走到$x$的点的集合中,哪一种颜色的点出现的次数最多。

   次数相同时输出编号最小的那个颜色。强制在线。

求哪种颜色可以用线段树合并搞定。

关键是这个强制在线。

当每次询问的时候,我们先要求出最小生成树在哪个时刻恰好把边权值不超过$w$的边都用并查集合并了。

在做最小生成树的时候每合并两个节点,另外开一个新的结点,原来两个点的父亲都指向这个新的结点。

然后倍增预处理,用类似求$LCA$的方法来得到询问的那个时刻。

时间复杂度$O(nlogn)$

#include <bits/stdc++.h>

using namespace std;

#define rep(i, a, b)	for (int i(a); i <= (b); ++i)
#define dec(i, a, b) for (int i(a); i >= (b); --i) const int N = 2e5 + 10;
const int M = N * 20; struct node{
int x, y, z;
void scan(){
scanf("%d%d%d", &x, &y, &z);
}
friend bool operator < (const node &a, const node &b){
return a.z < b.z;
}
} e[N]; int T, ca = 0;
int tot;
int n, m, q, ans;
int c[N], root[N], v[N], father[N];
int ls[M], rs[M], mx[M], ret[M];
int id, res[N];
int f[N][19]; int getfather(int x){
return father[x] == x ? x : father[x] = getfather(father[x]);
} void up(int i){
mx[i] = max(mx[ls[i]], mx[rs[i]]);
if (mx[i] == mx[ls[i]]) ret[i] = ret[ls[i]];
else ret[i] = ret[rs[i]];
} int build(int l, int r, int val){
int x = ++tot;
ls[x] = rs[x] = mx[x] = ret[x] = 0;
if (l == r){
mx[x] = 1;
ret[x] = val;
return x;
} int mid = (l + r) >> 1;
if (val <= mid) ls[x] = build(l, mid, val);
else rs[x] = build(mid + 1, r, val);
up(x);
return x;
} int Merge(int x, int y, int l, int r){
if (x == 0 || y == 0) return x + y;
if (l == r){
mx[x] += mx[y];
return x;
} int mid = (l + r) >> 1;
ls[x] = Merge(ls[x], ls[y], l, mid);
rs[x] = Merge(rs[x], rs[y], mid + 1, r);
up(x);
return x;
} int main(){ scanf("%d", &T);
while (T--){
tot = 0;
scanf("%d%d", &n, &m);
rep(i, 1, n) scanf("%d", c + i); rep(i, 1, n){
father[i] = i;
v[i] = 0;
f[i][0] = i;
root[i] = build(1, n, c[i]);
res[i] = ret[root[i]];
} id = n;
rep(i, 1, m) e[i].scan(); sort(e + 1, e + m + 1); rep(i, 1, m){
int x = e[i].x, y = e[i].y, z = e[i].z;
int fx = getfather(x), fy = getfather(y);
if (fx ^ fy){
++id;
f[id][0] = id;
father[id] = id;
v[id] = z;
father[fx] = father[fy] = id;
f[fx][0] = f[fy][0] = id;
root[id] = Merge(root[fx], root[fy], 1, n);
res[id] = ret[root[id]];
}
} rep(j, 1, 17){
rep(i, 1, id) f[i][j] = f[f[i][j - 1]][j - 1];
} printf("Case #%d:\n", ++ca);
scanf("%d", &q);
ans = 0;
while (q--){
int x, w;
scanf("%d%d", &x, &w);
x ^= ans, w ^= ans;
dec(i, 17, 0) if (v[f[x][i]] <= w) x = f[x][i];
printf("%d\n", ans = res[x]);
}
} return 0;
}

  

  

Codeforces Gym 101194G Pandaria (2016 ACM-ICPC EC-Final G题, 并查集 + 线段树合并)的更多相关文章

  1. ACM ICPC China final G Pandaria

    目录 ACM ICPC China final G Pandaria ACM ICPC China final G Pandaria 题意:给一张\(n\)个点\(m\)条边的无向图,\(c[i]\) ...

  2. Codeforces 571D - Campus(并查集+线段树+DFS 序,hot tea)

    Codeforces 题目传送门 & 洛谷题目传送门 看到集合的合并,可以本能地想到并查集. 不过这题的操作与传统意义上的并查集不太一样,传统意义上的并查集一般是用来判断连通性的,而此题还需支 ...

  3. Vladik and Entertaining Flags CodeForces - 811E (并查集,线段树)

    用线段树维护每一块左右两侧的并查集, 同色合并时若不连通则连通块数-1, 否则不变 #include <iostream> #include <algorithm> #incl ...

  4. 2016 ACM/ICPC Asia Regional Shenyang Online 1003/HDU 5894 数学/组合数/逆元

    hannnnah_j’s Biological Test Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 131072/131072 K ...

  5. 2016 ACM/ICPC Asia Regional Qingdao Online 1001/HDU5878 打表二分

    I Count Two Three Time Limit: 3000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others ...

  6. 2016 ACM/ICPC Asia Regional Shenyang Online 1009/HDU 5900 区间dp

    QSC and Master Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others) ...

  7. 2016 ACM/ICPC Asia Regional Shenyang Online 1007/HDU 5898 数位dp

    odd-even number Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)T ...

  8. 2016 ACM/ICPC Asia Regional Dalian Online 1002/HDU 5869

    Different GCD Subarray Query Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 65536/65536 K ( ...

  9. 2016 ACM/ICPC Asia Regional Dalian Online 1006 /HDU 5873

    Football Games Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)To ...

随机推荐

  1. Python面试题之一:解密

    Python面试题之一: 说明:就是Python工程师面试题 一.字典转换与正则提取值 1:key与Value交换 a = {'a':1,'b':2} print({value:key for key ...

  2. JS 如何获取radio或者checkbox选中后的值

    废话不多说,直接上代码: 代码: <!DOCTYPE html> <html> <head> <meta charset="UTF-8"& ...

  3. windows版本cloudbase-init流程说明

    源码流程说明 - 程序首先判断操作系统类型,加载对应的模块 - 加载服务,服务共分为四种: 'cloudbaseinit.metadata.services.httpservice.HttpServi ...

  4. windows下vim高亮systemverilog

    主要解决window环境下,vim高亮systemverilog的方法. 第一步:准备材料下载地址:https://files.cnblogs.com/files/aslmer/verilog_sys ...

  5. c++知识点总结--友元&运算符重载

    友元函数(不属于类) 可以访问类的私有变量,以及私有函数 友元函数在类内声明需要friend关键字,类外定义就不需要 友元函数可以直接在类内定义 友元函数必须包含对象指针   友元类(不适用继承,只适 ...

  6. Android记事本开发01

    今天: 学习一下Android的基本知识,了解一下记事本开发大概需要哪些知识. 昨天: 无 遇到的问题:

  7. C# 访问修饰符internal的访问范围误区释疑

      一.前言                                               MSDN关于访问修饰符的访问级别解释: 访问修饰符是一些关键字,用于指定声明的成员或类型的可访 ...

  8. 关于spark RDD trans action算子、lineage、宽窄依赖详解

    这篇文章想从spark当初设计时为何提出RDD概念,相对于hadoop,RDD真的能给spark带来何等优势.之前本想开篇是想总体介绍spark,以及环境搭建过程,但个人感觉RDD更为重要 铺垫 在h ...

  9. POJ 3907 Build Your Home | 计算多边形面积

    给个多边形 计算面积 输出要四舍五入 直接用向量叉乘就好 四舍五入可以+0.5向下取整 #include<cstdio> #include<algorithm> #includ ...

  10. BZOJ4031 [HEOI2015]小Z的房间 【矩阵树定理 + 高斯消元】

    题目链接 BZOJ4031 题解 第一眼:这不裸的矩阵树定理么 第二眼:这个模\(10^9\)是什么鬼嘛QAQ 想尝试递归求行列式,发现这是\(O(n!)\)的.. 想上高斯消元,却又处理不了逆元这个 ...