[ONTAK2010]Peaks
题目大意:
一个图上有$n(n\leq100000)$个带权点,$m(m\leq500000)$条带权边。有$q(q\leq500000)$组询问,每次询问从点$v$出发,只经过权值小于等于$x$的边能到达的点中,权值第$k$大的点权。
思路:
离线处理每个询问。将询问和边分别按权值排序,处理到当前询问时,将权值小于等于$x$的边加入到图中。用权值线段树维护每个连通块的权值。点的联通情况用并查集维护(相当于Kruskal),权值线段树直接合并。时间复杂度$O(n\log n+m\log m+q\log q+q\log n)$。
#include<cstdio>
#include<cctype>
#include<algorithm>
inline int getint() {
register char ch;
while(!isdigit(ch=getchar()));
register int x=ch^'';
while(isdigit(ch=getchar())) x=(((x<<)+x)<<)+(ch^'');
return x;
}
const int N=,M=,Q=;
int h[N],tmp[N],ans[Q];
struct Edge {
int u,v,w;
bool operator < (const Edge &another) const {
return w<another.w;
}
};
Edge e[M];
struct Query {
int v,x,k,id;
bool operator < (const Query &another) const {
return x<another.x;
}
};
Query q[Q];
struct DisjointSet {
int anc[N];
int find(const int &x) {
return x==anc[x]?x:anc[x]=find(anc[x]);
}
void reset(const int &n) {
for(register int i=;i<=n;i++) anc[i]=i;
}
void merge(const int &x,const int &y) {
anc[find(y)]=find(x);
}
bool same(const int &x,const int &y) {
return find(x)==find(y);
}
};
DisjointSet s;
class SegmentTree {
private:
struct Node {
int val;
Node *left,*right;
Node(const int &v):val(v),left(NULL),right(NULL) {}
};
void modify(Node *&p,const int &b,const int &e,const int &x) {
p=new Node();
if(b==e) return;
const int mid=(b+e)>>;
if(x<=mid) modify(p->left,b,mid,x);
if(x>mid) modify(p->right,mid+,e,x);
}
public:
Node *root[N];
void reset(const int &n) {
for(register int i=;i<=n;i++) {
modify(root[i],,tmp[],h[i]);
}
}
void merge(Node *&p1,Node *const &p2) {
if(!p1) {
p1=p2;
return;
}
if(!p2) return;
p1->val+=p2->val;
merge(p1->left,p2->left);
merge(p1->right,p2->right);
delete p2;
}
int query(const Node *const &p,const int &b,const int &e,const int &k) const {
if(p->val<k) return -;
if(b==e) return b;
const int mid=(b+e)>>;
if(!p->right) return query(p->left,b,mid,k);
if(k<=p->right->val) return query(p->right,mid+,e,k);
return query(p->left,b,mid,k-p->right->val);
}
};
SegmentTree t;
inline void add_edge(const Edge &e) {
if(s.same(e.u,e.v)) return;
t.merge(t.root[s.find(e.u)],t.root[s.find(e.v)]);
s.merge(e.u,e.v);
}
int main() {
const int n=getint(),m=getint(),cnt_q=getint();
for(register int i=;i<=n;i++) {
tmp[i]=h[i]=getint();
}
std::sort(&tmp[],&tmp[n]+);
tmp[]=std::unique(&tmp[],&tmp[n]+)-&tmp[];
for(register int i=;i<=n;i++) {
h[i]=std::lower_bound(&tmp[],&tmp[tmp[]]+,h[i])-&tmp[];
}
for(register int i=;i<m;i++) {
const int u=getint(),v=getint(),w=getint();
e[i]=(Edge){u,v,w};
}
std::sort(&e[],&e[m]);
for(register int i=;i<cnt_q;i++) {
const int v=getint(),x=getint(),k=getint();
q[i]=(Query){v,x,k,i};
}
std::sort(&q[],&q[cnt_q]);
s.reset(n);
t.reset(n);
for(register int i=,p=;i<cnt_q;i++) {
while(p<m&&e[p].w<=q[i].x) add_edge(e[p++]);
ans[q[i].id]=t.query(t.root[s.find(q[i].v)],,tmp[],q[i].k);
}
for(register int i=;i<cnt_q;i++) {
printf("%d\n",~ans[i]?tmp[ans[i]]:-);
}
return ;
}
[ONTAK2010]Peaks的更多相关文章
- bzoj 3545&&3551: [ONTAK2010]Peaks &&加强版 平衡树&&并查集合并树&&主席树
3545: [ONTAK2010]Peaks Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 635 Solved: 177[Submit][Stat ...
- BZOJ 3545: [ONTAK2010]Peaks( BST + 启发式合并 + 并查集 )
这道题很好想, 离线, 按询问的x排序从小到大, 然后用并查集维护连通性, 用平衡树维护连通块的山的权值, 合并就用启发式合并.时间复杂度的话, 排序是O(mlogm + qlogq), 启发式合并是 ...
- BZOJ 3551: [ONTAK2010]Peaks加强版 [Kruskal重构树 dfs序 主席树]
3551: [ONTAK2010]Peaks加强版 题意:带权图,多组询问与一个点通过边权\(\le lim\)的边连通的点中点权k大值,强制在线 PoPoQQQ大爷题解传送门 说一下感受: 容易发现 ...
- BZOJ 3545: [ONTAK2010]Peaks [Splay启发式合并]
3545: [ONTAK2010]Peaks 题意:带权图,多组询问与一个点通过边权\(\le x\)的边连通的点中点权k大值 又读错题了,输出点一直WA,问的是点权啊 本题加强版强制在线了,那这道题 ...
- bzoj3545: [ONTAK2010]Peaks 重构树 主席树
题目链接 bzoj3545: [ONTAK2010]Peaks 题解 套路重构树上主席树 代码 #include<cstdio> #include<algorithm> #de ...
- 【BZOJ3545】 [ONTAK2010]Peaks
BZOJ3545 [ONTAK2010]Peaks Solution 既然会加强版,直接把强制在线的操作去掉就好了. 代码实现 #include<stdio.h> #include< ...
- 【BZOJ3551】 [ONTAK2010]Peaks加强版
BZOJ3551 [ONTAK2010]Peaks加强版 Solution Kruscal重构树后发现可以对于小于的离散化然后倍增+主席树找到上一个的可行解. 然后就可以了. 如果数组开的不好,容易在 ...
- [ONTAK2010]Peaks kruskal重构树,主席树
[ONTAK2010]Peaks kruskal重构树练手题. LG传送门竟然不强制在线?看到离线水过很不爽:B站强制在线版传送门 看到"询问从点\(v\)开始只经过困难值小于等于\(x\) ...
- bzoj 3545/3551: [ONTAK2010]Peaks -- 主席树,最小生成树,倍增
3545: [ONTAK2010]Peaks Time Limit: 10 Sec Memory Limit: 128 MB Description 在Bytemountains有N座山峰,每座山峰 ...
- 【BZOJ3551】[ONTAK2010]Peaks加强版 最小生成树+DFS序+主席树
[BZOJ3545][ONTAK2010]Peaks Description 在Bytemountains有N座山峰,每座山峰有他的高度h_i.有些山峰之间有双向道路相连,共M条路径,每条路径有一个困 ...
随机推荐
- python偏函数使用
偏函数依托于python functools模块.
- 【转】git bash here 右键菜单失效后的修复方法
[HKEY_LOCAL_MACHINE\SOFTWARE\Classes\Directory\shell\git_shell] @="Git Ba&sh Here" [ ...
- Android开发——减小APK大小
0. 前言 APK的大小对APP的加载速度,使用内存大小和消耗功率多少有一定影响.如何减小APK的大小对于Android开发者是一个永恒的话题. 查阅了很多相关资料,并将其做了删减以及总结.本文原创, ...
- UVa 10118 记忆化搜索 Free Candies
假设在当前状态我们第i堆糖果分别取了cnt[i]个,那么篮子里以及口袋里糖果的个数都是可以确定下来的. 所以就可以使用记忆化搜索. #include <cstdio> #include & ...
- debug环境下打印
#ifdef DEBUG # define NSLog(...) NSLog(__VA_ARGS__) #else # define NSLog(...) {} #endif
- [转]mysql Access denied for user 'root'@'localhost' 问题的解决方法
解决方案如下: # /etc/init.d/mysql stop # mysqld_safe --user=mysql --skip-grant-tables --skip-networking &a ...
- WordPress后台添加侧边栏菜单
add_action('admin_menu', 'register_custom_menu_page'); function register_custom_menu_page() { add_me ...
- 二进制<1>
Matrix67:位运算简介及实用技巧(一) 基础篇 什么是位运算? 程序中的所有数在计算机内存中都是以二进制的形式储存的.位运算说穿了,就是直接对整数在内存中的二进制位进行操作.比如,and运 ...
- ACM程序设计选修课——Problem D: (ds:树)合并果子(最优二叉树赫夫曼算法)
Problem D: (ds:树)合并果子 Time Limit: 5 Sec Memory Limit: 64 MB Submit: 80 Solved: 4 [Submit][Status][ ...
- BZOJ4517 [Sdoi2016]排列计数 【组合数 + dp】
题目 求有多少种长度为 n 的序列 A,满足以下条件: 1 ~ n 这 n 个数在序列中各出现了一次 若第 i 个数 A[i] 的值为 i,则称 i 是稳定的.序列恰好有 m 个数是稳定的 满足条件的 ...