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序 分块】的更多相关文章

  1. HDU - 4366 Successor DFS序 + 分块暴力 or 线段树维护

    给定一颗树,每个节点都有忠诚和能力两个参数,随意指定一个节点,要求在它的子树中找一个节点代替它,这个节点要满足能力值大于它,而且是忠诚度最高的那个. 首先,dfs一下,处理出L[i], R[i]表示d ...

  2. 【HDU4366】【DFS序+分块】Successor

    Problem Description Sean owns a company and he is the BOSS.The other Staff has one Superior.every st ...

  3. HDU 4366 Successor(dfs序 + 分块)题解

    题意:每个人都有一个上司,每个人都有能力值和忠诚值,0是老板,现在给出m个询问,每次询问给出一个x,要求你找到x的所有直系和非直系下属中能力比他高的最忠诚的人是谁 思路:因为树上查询很麻烦,所以我们直 ...

  4. 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 ...

  5. BZOJ4867 Ynoi2017舌尖上的由乃(dfs序+分块)

    容易想到用dfs序转化为序列上的问题.考虑分块,对每块排序,修改时对于整块打上标记,边界暴力重构排序数组,询问时二分答案,这样k=sqrt(nlogn)时取最优复杂度nsqrt(nlogn)logn, ...

  6. BZOJ 4765 普通计算姬 dfs序+分块+树状数组+好题!!!

    真是道好题...感到灵魂的升华... 按dfs序建树状数组,拿前缀和去求解散块: 按点的标号分块,分成一个个区间,记录区间子树和 的 总和... 具体地,需要记录每个点u修改后,对每一个块i的贡献,记 ...

  7. bzoj 4765 普通计算姬 dfs序 + 分块

    题目链接 Description "奋战三星期,造台计算机".小G响应号召,花了三小时造了台普通计算姬.普通计算姬比普通计算机要厉害一些.普通计算机能计算数列区间和,而普通计算姬能 ...

  8. HDU 4366 Successor( DFS序+ 线段树 )

    Successor Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total S ...

  9. 2018.06.30 BZOJ4765: 普通计算姬(dfs序+分块+树状数组)

    4765: 普通计算姬 Time Limit: 30 Sec Memory Limit: 256 MB Description "奋战三星期,造台计算机".小G响应号召,花了三小时 ...

随机推荐

  1. RedHat6.1通过配置yum server安装软件包

    1.获取镜像RHEL_6.1\ x86_64\ Disc\ 1 2.配置yum server #cd /etc/yum.repos.d #tar -cvf 20141114bak.tar *.repo ...

  2. 【Java基础】枚举类与注解

    枚举类与注解 枚举类的使用 当需要定义一组常量时,强烈建议使用枚举类. 枚举类的理解:类的对象只有有限个,确定的. 若枚举只有一个对象, 则可以作为一种单例模式的实现方式. 枚举类的属性: 枚举类对象 ...

  3. linux之平均负载(学习笔记非原创)

    什么是平均负载 [root@111 ~]# uptime 11:03:33 up 149 days, 17:34, 1 user, load average: 0.08, 0.05, 0.01 最后三 ...

  4. 【Spring】Spring中的Bean - 4、Bean的生命周期

    Bean的生命周期 简单记录-Java EE企业级应用开发教程(Spring+Spring MVC+MyBatis)-Spring中的Bean 了解Spring中Bean的生命周期有何意义? 了解Sp ...

  5. LeetCode700. 二叉搜索树中的搜索

    题目 简单递归 1 class Solution { 2 public: 3 TreeNode* searchBST(TreeNode* root, int val) { 4 if(!root) re ...

  6. Apache目录详解

    Apache的主要目录和配置文件理解 参考链接:http://httpd.apache.org/docs/2.4/misc/security_tips.html 一.Apache主要配置文件注释(演示 ...

  7. Scrapy——將數據保存到MySQL數據庫

    Scrapy--將數據保存到MySQL數據庫 1. 在MySQL中創建數據庫表job_inf: 1 Create table job_inf( 2 id int(11) not null auto_i ...

  8. Arduino 上手实战呼吸灯

    前言 这篇稿子比以往的时候来的稍晚了一些,望fans们见谅,那即便如此,最终还是姗姗来迟了,公司新一轮战略性部署,被拖出去孵化新产品,开拓新市场去了,手头精力没有那么多了,另外产品一茬接一茬.韭菜一波 ...

  9. JAVA SSM整合流程以及注意点

    1.搭建整合环境 整合说明:SSM整合可以使用多种方式,咱们会选择XML + 注解的方式 先搭建整合的环境 先把Spring的配置搭建完成 再使用Spring整合SpringMVC框架 最后使用Spr ...

  10. 简话http请求

    一.http请求概念: 1.是指从客户端到服务器端的请求消息.包括:消息首行中,对资源的请求方法.资源的标识符及使用的协议. 包括请求(request)和响应(response) 2.过程: 域名解析 ...