传送门

一道考察比较全面的题。


这道题又用到了熟悉的kruskal+倍增来查找询问区间的方法。

查到询问的子树之后就可以用dfs序+主席树统计答案了。

代码:

#include<bits/stdc++.h>
#define N 200005
#define M 500005
using namespace std;
inline int read(){
	int ans=0;
	char ch=getchar();
	while(!isdigit(ch))ch=getchar();
	while(isdigit(ch))ans=(ans<<3)+(ans<<1)+(ch^48),ch=getchar();
	return ans;
}
inline void write(int x){
	if(x>9)write(x/10);
	putchar((x%10)^48);
}
int n,m,Q,h[N],val[N],fa[N],cnt=0,first[N],rt[N*3],num,son[N*25][2],siz[N*25],sz=0,st[N][20],w[N],mx[N][20],dep[N],q[N*3],bg[N],ed[N],top=0,lastans;
bool vis[N];
struct node{int u,v,w;}tt[M];
struct edge{int v,next;}e[N];
inline int find(int x){return x==fa[x]?x:fa[x]=find(fa[x]);}
inline int cmp(node a,node b){return a.w<b.w;}
inline void add(int u,int v){e[++cnt].v=v,e[cnt].next=first[u],first[u]=cnt;}
inline void update(int&p,int las,int l,int r,int v){
	p=++sz,siz[p]=siz[las]+1;
	if(l==r)return;
	son[p][0]=son[las][0],son[p][1]=son[las][1];
	int mid=l+r>>1;
	if(v<=mid)update(son[p][0],son[las][0],l,mid,v);
	else update(son[p][1],son[las][1],mid+1,r,v);
}
inline int query(int a,int b,int l,int r,int k){
	if(l==r)return l;
	int mid=l+r>>1;
	if(siz[son[b][0]]-siz[son[a][0]]>=k)return query(son[a][0],son[b][0],l,mid,k);
	return query(son[a][1],son[b][1],mid+1,r,k-siz[son[b][0]]+siz[son[a][0]]);
}
inline int Find(int x,int v){
	for(int i=17;~i;--i)if(dep[x]>=(1<<i)&&mx[x][i]<=v)x=st[x][i];
	return x;
}
inline void dfs(int p){
	q[++top]=p,vis[p]=1;
	for(int i=1;(1<<i)<=dep[p];++i)st[p][i]=st[st[p][i-1]][i-1],mx[p][i]=max(mx[p][i-1],mx[st[p][i-1]][i-1]);
	for(int i=first[p];i;i=e[i].next){
		int v=e[i].v;
		dep[v]=dep[p]+1,mx[v][0]=w[p],st[v][0]=p,dfs(v);
	}
	if(p>n)q[++top]=p;
}
inline void kruskal(){
	sort(tt+1,tt+m+1,cmp);
	for(int i=1;i<=n*2;++i)fa[i]=i;
	for(int i=1;i<=m;++i){
		int fx=find(tt[i].u),fy=find(tt[i].v);
		if(fx!=fy)fa[fx]=fa[fy]=++num,w[num]=tt[i].w,add(num,fx),add(num,fy);
		if(num==n*2-1)break;
	}
	for(int i=1;i<=n;++i)if(!vis[i])dfs(find(i));
	for(int i=1;i<=top;++i){
		int tmp=q[i];
		if(tmp<=n)update(rt[i],rt[i-1],1,n,h[tmp]);
		else{
			rt[i]=rt[i-1];
			if(!bg[tmp])bg[tmp]=i;
			else ed[tmp]=i;
		}
	}
}
int main(){
	n=read(),m=read(),Q=read(),lastans=-1,num=n;
	for(int i=1;i<=n;++i)val[i]=h[i]=read();
	sort(val+1,val+n+1);
	for(int i=1;i<=n;++i)h[i]=lower_bound(val+1,val+n+1,h[i])-val;
	for(int i=1;i<=m;++i)tt[i].u=read(),tt[i].v=read(),tt[i].w=read();
	kruskal();
	while(Q--){
		int v=read(),x=read(),k=read();
		if(~lastans)v^=lastans,x^=lastans,k^=lastans;
		int Rt=Find(v,x),ql=rt[bg[Rt]],qr=rt[ed[Rt]];
		if(siz[qr]-siz[ql]<k)puts("-1"),lastans=-1;
		else lastans=val[query(ql,qr,1,n,siz[qr]-siz[ql]-k+1)],printf("%d\n",lastans);

	}
	return 0;
}

2018.09.30 bzoj3551:Peaks加强版(dfs序+主席树+倍增+kruskal重构树)的更多相关文章

  1. BZOJ_3545_[ONTAK2010]Peaks_主席树+倍增+kruscal重构树+dfs序

    BZOJ_3545_[ONTAK2010]Peaks_主席树+倍增+kruscal重构树 Description 在Bytemountains有N座山峰,每座山峰有他的高度h_i.有些山峰之间有双向道 ...

  2. BZOJ 3551: [ONTAK2010]Peaks加强版 Kruskal重构树+dfs序+主席树+倍增

    建出来 $Kruskal$ 重构树. 将询问点向上跳到深度最小,且合法的节点上. 那么,得益于重构树优美的性质,这个最终跳到的点为根的所有子节点都可以与询问点互达. 对于子树中求点权第 $k$ 大的问 ...

  3. 2018.09.30 bzoj2223: [Coci 2009]PATULJCI(主席树)

    传送门 主席树经典题目. 直接利用主席树差分的思想判断区间中数的个数是否合法然后决定左走右走就行了. 实际上跟bzoj3524是同一道题. 代码: #include<bits/stdc++.h& ...

  4. 【bzoj3545/bzoj3551】[ONTAK2010]Peaks/加强版 Kruskal+树上倍增+Dfs序+主席树

    bzoj3545 题目描述 在Bytemountains有N座山峰,每座山峰有他的高度h_i.有些山峰之间有双向道路相连,共M条路径,每条路径有一个困难值,这个值越大表示越难走,现在有Q组询问,每组询 ...

  5. dfs序+主席树 或者 树链剖分+主席树(没写) 或者 线段树套线段树 或者 线段树套splay 或者 线段树套树状数组 bzoj 4448

    4448: [Scoi2015]情报传递 Time Limit: 20 Sec  Memory Limit: 256 MBSubmit: 588  Solved: 308[Submit][Status ...

  6. 【bzoj1803】Spoj1487 Query on a tree III DFS序+主席树

    题目描述 You are given a node-labeled rooted tree with n nodes. Define the query (x, k): Find the node w ...

  7. 【BZOJ3551】[ONTAK2010]Peaks加强版 最小生成树+DFS序+主席树

    [BZOJ3545][ONTAK2010]Peaks Description 在Bytemountains有N座山峰,每座山峰有他的高度h_i.有些山峰之间有双向道路相连,共M条路径,每条路径有一个困 ...

  8. BZOJ 3551: [ONTAK2010]Peaks加强版 [Kruskal重构树 dfs序 主席树]

    3551: [ONTAK2010]Peaks加强版 题意:带权图,多组询问与一个点通过边权\(\le lim\)的边连通的点中点权k大值,强制在线 PoPoQQQ大爷题解传送门 说一下感受: 容易发现 ...

  9. BZOJ3551 ONTAK2010Peaks加强版(kruskal重构树+dfs序+主席树)

    kruskal重构树本质就是给并查集显式建树来替代可持久化并查集.将边按困难度从小到大排序后建出该树,按dfs序建主席树即可.查询时跳到深度最浅的满足在该重要度下已被合并的点,在子树内查询第k大. # ...

随机推荐

  1. Devexpress 百分号显示格式

    百分号:{0:P}表示显示为百分号模式.如数据源中为0.5.表示出来为50%

  2. 机器学习入门-DBSCAN聚类算法

    DBSCAN 聚类算法又称为密度聚类,是一种不断发张下线而不断扩张的算法,主要的参数是半径r和k值 DBSCAN的几个概念: 核心对象:某个点的密度达到算法设定的阈值则其为核心点,核心点的意思就是一个 ...

  3. Flex 布局教程:实例

    分类: 开发者手册 Flex 布局教程:实例篇   作者: 阮一峰 日期: 2015年7月14日 上一篇文章介绍了Flex布局的语法,今天介绍常见布局的Flex写法. 你会看到,不管是什么布局,Fle ...

  4. 值得推荐的C/C++开源框架和库

    值得推荐的C/C++开源框架和库  转自:http://www.cnblogs.com/lidabo/p/5514155.html   - 1. Webbench Webbench是一个在Linux下 ...

  5. ntohs, ntohl, htons,htonl的比较和详解【转】

    ntohs =net to host short int 16位 htons=host to net short int 16位 ntohs =net to host long int 32位 hto ...

  6. Ubuntu下启动 Redis时, 提示 "Can't open the log file: Permission denied failed"

    问题来源:在删除var目录下的log文件时,将redis文件夹删除了.然后在重启时:/etc/init.d/redis-server start,提示: Starting redis-server: ...

  7. python的可变list和不可变tuple, dict和set

    list和tuple 在python中分为可变表和不可变表: 类型 名称 表示方法 可变 list [] 不可变 tuple () list list是可变表,list内部索引从0开始,正整数是正序的 ...

  8. 常用类一一MATH类一一两个静态常量PI 和E,一些数学函数。

    package test; public class MathTest { public static void main(String[] args) { System.out.println(Ma ...

  9. go递归函数如何传递数组切片slice

    数组切片slice这个东西看起来很美好,真正用起来会发现有诸多的不爽. 第一,数组.数组切片混淆不清,使用方式完全一样,有时候一些特性又完全不一样,搞不清原理很容易误使用. 第二,数组切片的appen ...

  10. springboot不能加载*.properties

    代码检查了无数遍!这是在intellij IDEA 下!!! 如图,这些配置文件直接放在sms-server下面,并没有放在sms-server/src/main/resources下面,所以不是cl ...