\(\\\)

Description


其实这题才是正版的 Qtree3......

给定 \(n\) 个点,以 \(1\) 号节点为根的树,点有点权。

\(m\) 次询问 以 \(x\) 为根的子树内,点权第 \(k\) 小的 节点编号 是多少。

有多组测试数据,每组数据以 \(DONE\) 结尾。

  • \(n,m\le 10^5\)

\(\\\)

Solution


注意到一棵树的子树 \(DFS\) 序是连续的。

一遍 \(DFS\) 确定 \(DFS\) 序以及各个点的子树大小。

在 \(DFS\) 序上建主席树就可以了。

查子树查的其实就是 \((dfn[x]-1,dfn[x]+size[x]-1]\) 。

注意查的是点的编号,但是注意到点权两两不同,就可以离散化后记录每一个点权对应的节点编号了。

主席树注意空间 开成17倍成功RE

\(\\\)

Code


#include<cmath>
#include<cstdio>
#include<cctype>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#define N 100010
#define gc getchar
#define Rg register
#define mid ((l+r)>>1)
using namespace std; inline int rd(){
int x=0; bool f=0; char c=gc();
while(!isdigit(c)){if(c=='-')f=1;c=gc();}
while(isdigit(c)){x=(x<<1)+(x<<3)+(c^48);c=gc();}
return f?-x:x;
} int n,m,tot,hd[N],val[N],tmp[N],len; struct edge{int to,nxt;}e[N<<1]; inline void add(int u,int v){
e[++tot].to=v; e[tot].nxt=hd[u]; hd[u]=tot;
} int cnt,dfn[N],sz[N],vpos[N],pos[N]; void dfs(int u,int fa){
sz[u]=1;
dfn[u]=++cnt;
vpos[cnt]=val[u];
for(Rg int i=hd[u],v;i;i=e[i].nxt)
if((v=e[i].to)!=fa){dfs(v,u);sz[u]+=sz[v];}
} struct seg{ int rot[N],ptr; inline int newnode(){return ++ptr;} struct node{int ls,rs,sum;}c[N*20]; void build(int &rt,int l,int r){
rt=newnode();
if(l==r) return;
build(c[rt].ls,l,mid);
build(c[rt].rs,mid+1,r);
} inline void pushup(int rt){
c[rt].sum=c[c[rt].ls].sum+c[c[rt].rs].sum;
} void insert(int &rt,int lst,int l,int r,int x){
rt=newnode();
if(l==r){c[rt].sum=c[lst].sum+1;return;}
if(x<=mid){
c[rt].rs=c[lst].rs;
insert(c[rt].ls,c[lst].ls,l,mid,x);
}
else{
c[rt].ls=c[lst].ls;
insert(c[rt].rs,c[lst].rs,mid+1,r,x);
}
pushup(rt);
} inline int query(int rtl,int rtr,int l,int r,int k){
if(l==r) return l;
int nowans=c[c[rtr].ls].sum-c[c[rtl].ls].sum;
if(k<=nowans) return query(c[rtl].ls,c[rtr].ls,l,mid,k);
else return query(c[rtl].rs,c[rtr].rs,mid+1,r,k-nowans);
} }tree; int main(){
n=rd();
for(Rg int i=1;i<=n;++i) tmp[i]=val[i]=rd();
sort(tmp+1,tmp+1+n);
for(Rg int i=1;i<=n;++i){
tmp[++len]=tmp[i];
while(tmp[i+1]==tmp[i]) ++i;
}
for(Rg int i=1;i<=n;++i){
val[i]=lower_bound(tmp+1,tmp+1+len,val[i])-tmp;
pos[val[i]]=i;
}
for(Rg int i=1,u,v;i<n;++i){
u=rd(); v=rd(); add(u,v); add(v,u);
}
dfs(1,0);
tree.build(tree.rot[0],1,len);
for(Rg int i=1;i<=n;++i) tree.insert(tree.rot[i],tree.rot[i-1],1,len,vpos[i]);
m=rd();
for(Rg int i=1,rt,k;i<=m;++i){
rt=rd(); k=rd();
printf("%d\n",pos[tree.query(tree.rot[dfn[rt]-1],tree.rot[dfn[rt]+sz[rt]-1],1,len,k)]);
}
return 0;
}

[ SPOJ PT07J ] Query on a tree III的更多相关文章

  1. SPOJ PT07J - Query on a tree III(划分树)

    PT07J - Query on a tree III #tree You are given a node-labeled rooted tree with n nodes. Define the ...

  2. SP1487 PT07J - Query on a tree III (主席树)

    SP1487 PT07J - Query on a tree III 题意翻译 你被给定一棵带点权的n个点的有根数,点从1到n编号. 定义查询 query(x,k): 寻找以x为根的k大点的编号(从小 ...

  3. SPOJ 1487 Query on a tree III(划分树)

    题目链接:http://www.spoj.com/problems/PT07J/ 题意:给出一个有根树,1为根节点,每个节点有权值.若干询问,询问以u为根的子树中权值第K小的节点编号. 思路:DFS一 ...

  4. SP1487 PT07J - Query on a tree III 主席树+dfs序

    Code: #include<iostream> #include<cstdio> #include<algorithm> #include<string&g ...

  5. 【BZOJ1803】Spoj1487 Query on a tree III 主席树+DFS序

    [BZOJ1803]Spoj1487 Query on a tree III Description You are given a node-labeled rooted tree with n n ...

  6. 「SPOJ1487」Query on a tree III

    「SPOJ1487」Query on a tree III 传送门 把树的 \(\text{dfs}\) 序抠出来,子树的节点的编号位于一段连续区间,然后直接上建主席树区间第 \(k\) 大即可. 参 ...

  7. SPOJ 375. Query on a tree (树链剖分)

    Query on a tree Time Limit: 5000ms Memory Limit: 262144KB   This problem will be judged on SPOJ. Ori ...

  8. SPOJ QTREE Query on a tree 树链剖分+线段树

    题目链接:http://www.spoj.com/problems/QTREE/en/ QTREE - Query on a tree #tree You are given a tree (an a ...

  9. QTREE3 spoj 2798. Query on a tree again! 树链剖分+线段树

    Query on a tree again! 给出一棵树,树节点的颜色初始时为白色,有两种操作: 0.把节点x的颜色置反(黑变白,白变黑). 1.询问节点1到节点x的路径上第一个黑色节点的编号. 分析 ...

随机推荐

  1. easyUi 学习笔记 (一) 使用easyui 和ztree 创建前端框架

    <%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding= ...

  2. sdfs

    <!DOCTYPE html><html><head><meta charset="GB18030"><title>In ...

  3. Lua中..和#运算符的用法

    Lua中..和#运算符的用法 样例 试试以下的样例就明确了在Lua编程语言提供的其它运算符: a = "Hello " b = "World" print(&q ...

  4. ViewPager学习之仿微信主界面

    由于素材的原因,这里都是从网上找的图片,所以所谓的仿微信实际上最后成了下图这货.. .,点击变色也是自己用的windows自带绘图的颜料桶填充的空白. .. watermark/2/text/aHR0 ...

  5. CentOS+Apache+php无法访问redis的解决方法 Redis server went away

    在CentOS下配置Apache+php+redis+phpredis环境.编辑访问redis缓存的php程序test.php,以应用程序方式在后台运行,可成功访问Redis,而在Apache下以网页 ...

  6. train_action

    # 导入数值计算模块 import numpy as np import tensorflow as tf # 创建计算会话 sess = tf.Session() # 生成数据,创建占位符和变量A ...

  7. oracle存储过程和游标的使用

    oracle存储过程和游标的使用 (2011-04-19 14:52:47) 转载▼ 游标: 用来查询数据库,获取记录集合(结果集)的指针,我们所说的游标通常是指显式游标,因此从现在起没有特别指明的情 ...

  8. 分布式消息中间件Rabbit Mq的了解与使用

    MQ(消息队列)作为现代比较流行的技术,在互联网应用平台中作为中间件,主要解决了应用解耦.异步通信.流量削锋.服务总线等问题,为实现高并发.高可用.高伸缩的企业应用提供了条件. 目前市面比较流行的消息 ...

  9. Git-flow 一个简单高效的Git工作流

    背景 由于Git的分支比SVN更好管理且更易使用,最近团队从SVN迁移到Git,需要重新规划开发流程,最终确定使用Git-flow工作流,这是目前比较流行的一种分支模型,下面是Git-flow的简易流 ...

  10. C# 简单实现直线方程,抛物线方程(转载)

    http://www.cnblogs.com/hsiang/archive/2017/01/17/6294864.html 本例子是简单的在WinForm程序中实现在坐标系中绘制直线方程,抛物线方程, ...