3551: [ONTAK2010]Peaks加强版

Time Limit: 20 Sec  Memory Limit: 128 MB
Submit: 2438  Solved: 763
[Submit][Status][Discuss]

Description

【题目描述】同3545

Input

第一行三个数N,M,Q。
第二行N个数,第i个数为h_i
接下来M行,每行3个数a b c,表示从a到b有一条困难值为c的双向路径。
接下来Q行,每行三个数v x k,表示一组询问。v=v xor lastans,x=x xor lastans,k=k xor lastans。如果lastans=-1则不变。
 

Output

同3545

Sample Input

 

Sample Output

 

HINT

【数据范围】同3545

Source

[Submit][Status][Discuss]

做一遍Kruskal,每次加入一条边的时候为这条边新建一个点,点权为其边权,分别连向两个连通块。就建出了一个Kruskal重构树。

重构树的性质:是一个二叉堆,且原图中两点的满足“路径上最大边权最小”的路径的最大边权就是新树中两点的LCA的点权。

建出来求出DFS序跑主席树即可。

接下来吐槽这道题。题目是3545的加强版,加了强制在线不给样例,结果拿原题的样例死活测不对。导致我对着一个正确的程序调了整整一个下午。

#include<cstdio>
#include<algorithm>
#define rep(i,l,r) for (int i=l; i<=r; i++)
using namespace std; const int N=,M=;
int n,m,Q,x,ans,val[N],tot,u,lim,k,tim,cnt,nd,dep[N],dfn[N],L[N],R[N],ls[M],rs[M],sz[M];
int fa[N][],mp[N],f[N],to[N<<],nxt[N<<],h[N],root[N];
struct P{ int u,v,w; bool operator <(const P &r)const{ return w<r.w; }}a[];
int find(int x){ return (x==f[x]) ? x : f[x]=find(f[x]); }
void add(int u,int v){ to[++cnt]=v; nxt[cnt]=h[u]; h[u]=cnt; } void Kruskal(){
sort(a+,a+m+); int k=;
rep(i,,m){
int u=a[i].u,v=a[i].v,w=a[i].w,x=find(u),y=find(v);
if (x==y) continue;
val[++tot]=w; add(tot,x); add(tot,y); f[x]=f[y]=f[tot]=tot;
if (++k==n-) break;
}
} void dfs(int u){
if (u<=n) dfn[++tim]=u; L[u]=tim;
for (int i=; (<<i)<=dep[u]; i++) fa[u][i]=fa[fa[u][i-]][i-];
for (int i=h[u],k; i; i=nxt[i])
if ((k=to[i])!=fa[u][]) fa[k][]=u,dep[k]=dep[u]+,dfs(k);
R[u]=tim;
} void insert(int y,int &x,int L,int R,int pos){
ls[x=++nd]=ls[y]; rs[x]=rs[y]; sz[x]=sz[y]+;
if (L==R) return; int mid=(L+R)>>;
if (pos<=mid) insert(ls[y],ls[x],L,mid,pos); else insert(rs[y],rs[x],mid+,R,pos);
} int que(int x,int y,int k){
x=root[x]; y=root[y]; int all=sz[y]-sz[x];
if (k>all) return -;
k=all-k+; int L=,R=*mp;
while (L!=R){
int S=sz[ls[y]]-sz[ls[x]],mid=(L+R)>>;
if (k<=S) x=ls[x],y=ls[y],R=mid;
else x=rs[x],y=rs[y],L=mid+,k-=S;
}
return L;
} int main(){
freopen("bzoj3551.in","r",stdin);
freopen("bzoj3551.out","w",stdout);
scanf("%d%d%d",&n,&m,&Q); tot=n;
rep(i,,n) scanf("%d",&x),val[i]=mp[i]=x,f[i]=i;
val[]=1e9+; rep(i,,m) scanf("%d%d%d",&a[i].u,&a[i].v,&a[i].w);
Kruskal(); dfs(tot); sort(mp+,mp+n+); mp[]=unique(mp+,mp+n+)-mp-;
rep(i,,n) val[i]=lower_bound(mp+,mp+mp[]+,val[i])-mp;
rep(i,,n) insert(root[i-],root[i],,*mp,val[dfn[i]]);
rep(i,,Q){
if (ans==-) ans=;
scanf("%d%d%d",&u,&lim,&k); u^=ans; lim^=ans; k^=ans;
for (int i=; ~i; i--) if (val[fa[u][i]]<=lim) u=fa[u][i];
ans=que(L[u],R[u],k);
if (~ans) ans=mp[ans]; printf("%d\n",ans);
}
return ;
}

[BZOJ3551][ONTAK2010]Peaks(加强版)(Kruskal重构树,主席树)的更多相关文章

  1. BZOJ.3551.[ONTAK2010]Peaks加强版(Kruskal重构树 主席树)

    题目链接 \(Description\) 有n个座山,其高度为hi.有m条带权双向边连接某些山.多次询问,每次询问从v出发 只经过边权<=x的边 所能到达的山中,第K高的是多少. 强制在线. \ ...

  2. 【BZOJ-3545&3551】Peaks&加强版 Kruskal重构树 + 主席树 + DFS序 + 倍增

    3545: [ONTAK2010]Peaks Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1202  Solved: 321[Submit][Sta ...

  3. luoguP4197:Peaks(Kruskal重构树+主席树)或者(点分树+离线)

    题意:有N座山,M条道路.山有山高,路有困难值(即点权和边权).现在Q次询问,每次给出(v,p),让求从v出发,只能结果边权<=p的边,问能够到达的山中,第K高的高度(从大到小排序). 思路:显 ...

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

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

  5. 【BZOJ 3551】[ONTAK2010] Peaks加强版 Kruskal重构树+树上倍增+主席树

    这题真刺激...... I.关于Kruskal重构树,我只能开门了,不过补充一下那玩意还是一棵满二叉树.(看一下内容之前请先进门坐一坐) II.原来只是用树上倍增求Lca,但其实树上倍增是一种方法,L ...

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

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

  7. BZOJ3551 Peaks加强版 [Kruskal重构树,主席树]

    BZOJ 思路 我觉得这题可持久化线段树合并也可以做 我觉得这题建出最小生成树之后动态点分治+线段树也可以做 还是学习一下Kruskal重构树吧-- Kruskal重构树,就是在做最小生成树的时候,如 ...

  8. BZOJ3545&3551[ONTAK2010]Peaks——kruskal重构树+主席树+dfs序+树上倍增

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

  9. 【BZOJ3545&BZOJ3551】Peaks(kruskal重构树,主席树,dfs序)

    题意:在Bytemountains有N座山峰,每座山峰有他的高度h_i. 有些山峰之间有双向道路相连,共M条路径,每条路径有一个困难值,这个值越大表示越难走, 现在有Q组询问,每组询问询问从点v开始只 ...

随机推荐

  1. 【NOIP模拟赛】黑红树 期望概率dp

    这是一道比较水的期望概率dp但是考场想歪了.......我们可以发现奇数一定是不能掉下来的,因为若奇数掉下来那么上一次偶数一定不会好好待着,那么我们考虑,一个点掉下来一定是有h/2-1个红(黑),h/ ...

  2. Codeforces ----- Kefa and Dishes [状压dp]

    题目传送门:580D 题目大意:给你n道菜以及每道菜一个权值,k个条件,即第y道菜在第x道后马上吃有z的附加值,求从中取m道菜的最大权值 看到这道题,我们会想到去枚举,但是很显然这是会超时的,再一看数 ...

  3. 怎么给word加底纹

  4. js中连写两个?:三元运算符语法解释

    在angular 源码中有连写两个三元运算符的代码: var hash = isString(hash) ? hash : isNumber(hash) ? hash.toString() :$loc ...

  5. Web自适应

    随着移动设备的普及,移动web在前端工程师们的工作中占有越来越重要的位置.移动设备更新速度频繁,手机厂商繁多,导致的问题是每一台机器的屏幕宽度和分辨率不一样.这给我们在编写前端界面时增加了困难,适配问 ...

  6. python 闭包与装饰器

    1.闭包--返回子函数名 作用:使用子函数之外的父函数的变量 闭包就是你调用了一个函数a,这个函数a反悔了一个子函数名b,这个返回的函数b就叫做闭包 代码举例 def a(): test = 'aa' ...

  7. 谈pkusc2016的几道数学题

    题面搬来的qwq(忘记出处了 水印应该能表示) [题解] 1. 我们看到这题先想到令(x+y+z)^3 展开得到一坨,稍微减减,得到我们要求证 delta = 3xy^2+3xz^2+3yx^2+3y ...

  8. COGS2090 Asm.Def找燃料

    时间限制:1 s   内存限制:256 MB [题目描述] “听说咱们要完了?”比利·海灵顿拨弄着操纵杆,头也不回地问Asm.Def. “不要听得风就是雨.” “开个玩笑嘛.不就是打机器人,紧张啥,你 ...

  9. 51nod 1791 合法括号子段

    有一个括号序列,现在要计算一下它有多少非空子段是合法括号序列. 合法括号序列的定义是: 1.空序列是合法括号序列. 2.如果S是合法括号序列,那么(S)是合法括号序列.3.如果A和B都是合法括号序列, ...

  10. 【洛谷 P1666】 前缀单词 (Trie)

    题目链接 考试时暴搜50分...其实看到"单词","前缀"这种字眼时就要想到\(Trie\)的,哎,我太蒻了. 以一个虚点为根,建一棵\(Trie\),然后\( ...