HDU4366 Successor【dfs序 分块】
HDU4366 Successor
题意:
给出一棵根为\(1\)的树,每个点有两个权值\(x,y\),每次询问一个点的子树中\(x\)比这个点的\(x\)大且\(y\)值最大的那个点
题解:
如果以dfs序来看的话,每个点的子树可以看作是dfs序的一段区间
然后我们对这个序列分块,每个块内按\(x\)排序,然后维护后缀\(y\)的最大值
每次查询的时候对于一个块,可以先二分出来符合条件的\(x\)最小的那个位置,然后找后缀\(y\)最大的那个点
如果要单点修改的话,每次对一个块暴力修改也是没有问题的
//#pragma GCC optimize("O3")
//#pragma comment(linker, "/STACK:1024000000,1024000000")
#include<bits/stdc++.h>
using namespace std;
function<void(void)> ____ = [](){ios_base::sync_with_stdio(false); cin.tie(0); cout.tie(0);};
const int MAXN = 5e4+7;
int T,n,m,q;
vector<int> G[MAXN];
int st[MAXN],ed[MAXN],belong[MAXN],l[MAXN],r[MAXN],sqt;
pair<int,int> emp[MAXN];
vector<int> ord;
void dfs(int u){
st[u] = ord.size();
ord.push_back(u);
for(int v : G[u]) dfs(v);
ed[u] = ord.size() - 1;
}
void divide(){
sqt = sqrt(n);
m = n / sqt + (n%sqt==0?0:1);
for(int i = 0; i < n; i++) belong[i] = i / sqt + 1;
for(int i = 1; i <= m; i++) l[i] = sqt * (i-1), r[i] = sqt * i - 1;
r[m] = n - 1;
}
vector<pair<pair<int,int>,int>> vec[MAXN];
vector<pair<int,int>> suf[MAXN];
void build(){
for(int i = 1; i <= m; i++){
vec[i].clear(); suf[i].clear();
for(int j = l[i]; j <= r[i]; j++) vec[i].emplace_back(make_pair(emp[ord[j]],ord[j]));
sort(vec[i].begin(),vec[i].end());
suf[i].resize(vec[i].size());
suf[i].back() = make_pair(vec[i].back().first.second,vec[i].back().second);
for(int j = ((int)suf[i].size()) - 2; j >= 0; j--){
if(vec[i][j].first.second>suf[i][j+1].first) suf[i][j] = make_pair(vec[i][j].first.second,vec[i][j].second);
else suf[i][j] = suf[i][j+1];
}
}
}
void ask(){
int x;
scanf("%d",&x); x++;
int L = st[x], R = ed[x];
x = emp[x].first;
int lp = belong[L], rp = belong[R];
int res = -1;
if(lp==rp){
for(int i = L; i <= R; i++){
if(emp[ord[i]].first<=x) continue;
if(res==-1) res = ord[i];
else if(emp[ord[i]].second>emp[res].second) res = ord[i];
}
}
else{
for(int i = L; i <= r[lp]; i++){
if(emp[ord[i]].first<=x) continue;
if(res==-1) res = ord[i];
else if(emp[ord[i]].second>emp[res].second) res = ord[i];
}
for(int i = l[rp]; i <= R; i++){
if(emp[ord[i]].first<=x) continue;
if(res==-1) res = ord[i];
else if(emp[ord[i]].second>emp[res].second) res = ord[i];
}
lp++, rp--;
for(int i = lp; i <= rp; i++){
auto p = lower_bound(vec[i].begin(),vec[i].end(),make_pair(make_pair(x+1,0),1));
if(p==vec[i].end()) continue;
int pt = p - vec[i].begin();
if(res==-1 or suf[i][pt].first>emp[res].second) res = suf[i][pt].second;
}
}
printf("%d\n",res==-1?-1:res-1);
}
void solve(){
scanf("%d %d",&n,&q);
for(int i = 1; i <= n; i++) G[i].clear();
for(int i = 2; i <= n; i++){
int x;
scanf("%d %d %d",&x,&emp[i].second,&emp[i].first);
G[++x].push_back(i);
}
ord.clear();
dfs(1);
divide();
build();
while(q--) ask();
}
int main(){
for(scanf("%d",&T); T; T--) solve();
return 0;
}
HDU4366 Successor【dfs序 分块】的更多相关文章
- HDU - 4366 Successor DFS序 + 分块暴力 or 线段树维护
给定一颗树,每个节点都有忠诚和能力两个参数,随意指定一个节点,要求在它的子树中找一个节点代替它,这个节点要满足能力值大于它,而且是忠诚度最高的那个. 首先,dfs一下,处理出L[i], R[i]表示d ...
- 【HDU4366】【DFS序+分块】Successor
Problem Description Sean owns a company and he is the BOSS.The other Staff has one Superior.every st ...
- HDU 4366 Successor(dfs序 + 分块)题解
题意:每个人都有一个上司,每个人都有能力值和忠诚值,0是老板,现在给出m个询问,每次询问给出一个x,要求你找到x的所有直系和非直系下属中能力比他高的最忠诚的人是谁 思路:因为树上查询很麻烦,所以我们直 ...
- 2016 ACM/ICPC Asia Regional Dalian Online 1010 Weak Pair dfs序+分块
Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 262144/262144 K (Java/Others)Total Submissio ...
- BZOJ4867 Ynoi2017舌尖上的由乃(dfs序+分块)
容易想到用dfs序转化为序列上的问题.考虑分块,对每块排序,修改时对于整块打上标记,边界暴力重构排序数组,询问时二分答案,这样k=sqrt(nlogn)时取最优复杂度nsqrt(nlogn)logn, ...
- BZOJ 4765 普通计算姬 dfs序+分块+树状数组+好题!!!
真是道好题...感到灵魂的升华... 按dfs序建树状数组,拿前缀和去求解散块: 按点的标号分块,分成一个个区间,记录区间子树和 的 总和... 具体地,需要记录每个点u修改后,对每一个块i的贡献,记 ...
- bzoj 4765 普通计算姬 dfs序 + 分块
题目链接 Description "奋战三星期,造台计算机".小G响应号召,花了三小时造了台普通计算姬.普通计算姬比普通计算机要厉害一些.普通计算机能计算数列区间和,而普通计算姬能 ...
- HDU 4366 Successor( DFS序+ 线段树 )
Successor Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others)Total S ...
- 2018.06.30 BZOJ4765: 普通计算姬(dfs序+分块+树状数组)
4765: 普通计算姬 Time Limit: 30 Sec Memory Limit: 256 MB Description "奋战三星期,造台计算机".小G响应号召,花了三小时 ...
随机推荐
- Unity优化图解
花了2天把之前学到的一些关于优化的知识全都写了下来,放到一张表里面 https://www.processon.com/mindmap/5cf64f53e4b0bc8329e8112e
- LeetCode105 从前序和中序序列构造二叉树
题目描述: 根据一棵树的前序遍历与中序遍历构造二叉树. 注意:你可以假设树中没有重复的元素. 例如,给出 前序遍历 preorder = [3,9,20,15,7] 中序遍历 inorder = [9 ...
- LeetCode141-环形链表检测
题目 给定一个链表,判断链表中是否有环. 如果链表中有某个节点,可以通过连续跟踪 next 指针再次到达,则链表中存在环. 为了表示给定链表中的环,我们使用整数 pos 来表示链表尾连接到链表中的位置 ...
- 【Linux】以001格式循环到100保证位数是3位
这里有一个前提,要保证数位是相同的 确实数字是1-100 但是数位是不同的,需要统一一下位数必须是3位的 这个问题在很多论坛上用的都是printf这个命令,确实可以达到这个效果,但是没有我下面介绍的 ...
- kubectl命令管理
kubectl命令管理 查看更多帮助命令 [root@k8s-master ~]# kubectl --help 创建一个命名空间 [root@k8s-master ~]# kubectl creat ...
- [从源码学设计]蚂蚁金服SOFARegistry之配置信息
[从源码学设计]蚂蚁金服SOFARegistry之配置信息 目录 [从源码学设计]蚂蚁金服SOFARegistry之配置信息 0x00 摘要 0x01 业务范畴 1.1 配置作用 1.2 学习方向 0 ...
- Centos 7 关机和重启 命令
1,关机命令 1 shutdown -h now/0 2 halt 3 init 0 4 poweroff 5 举例: 6 shutdown -h 3 ------3分钟后关机(可用shutdown ...
- uni-app开发经验分享十二: Android平台应用启动时读写手机存储、访问设备信息(如IMEI)等权限策略及提示信息
Android平台从6.0(API23)开始系统对权限的管理更加严格,所有涉及敏感权限都需要用户授权允许才能获取.因此一些应用基础业务逻辑需要的权限会在应用启动时申请,并引导用户允许. 读写手机存储权 ...
- spring源码分析之玩转ioc:bean初始化和依赖注入(一)
最近赶项目,天天加班到十一二点,终于把文档和代码都整完了,接上继续整. 上一篇聊了beanProcess的注册以及对bean的自定义修改和添加,也标志着创建bean的准备工作都做好了,接下来就是开大招 ...
- kvm虚拟机管理(创建、连接)
创建虚机.远程管理kvm虚机.virsh命令行下管理虚机..kvm通过virsh console 连入虚拟机 一.创建虚机 1)打开虚拟化管理器