[Ynoi2012]D1T3
https://www.luogu.org/problemnew/show/P5311
题解
先把点分树建出来。
对于吗,每一个询问\((l,r,x)\),我们对于x要找到它在点分树上最靠上的父亲节点使得两点之间的点在\(l \sim r\)中。
然后问题就变成了从一个根出发,可以经过\(l \sim r\)的点,能访问的颜色个数。
那么因为有了点分树,所以我们对于每个点\(dfs\)它的子树的复杂度是对的。
然后对于每个根,我们求出它的子树里的所有点的到根的路径上的编号最大最小值,然后考虑这个点产生贡献的条件,发现就是个二维偏序。
但是还有相同颜色算两边的情况,这个在扫描线的时候用单调性去重就好了,具体来说对于\(r\)都在合法范围内的点,\(l\)更大越优,所以我们只需要保留最大的\(l\)就好了。
代码
#include<bits/stdc++.h>
#define N 100009
#define mm make_pair
using namespace std;
typedef long long ll;
vector<int>fa[N];
vector<pair<int,int> >v[N];
bool vis[N];
int tot,head[N],dp[N],ans[N],size[N],root,tong[N],a[N],n,m,sum;
inline ll rd(){
ll x=0;char c=getchar();bool f=0;
while(!isdigit(c)){if(c=='-')f=1;c=getchar();}
while(isdigit(c)){x=(x<<1)+(x<<3)+(c^48);c=getchar();}
return f?-x:x;
}
struct edge{
int n,to;
}e[N<<1];
struct BIT{
int tr[N];
inline void add(int x,int y){while(x)tr[x]+=y,x-=x&-x;}
inline int query(int x){int ans=0;while(x<=n)ans+=tr[x],x+=x&-x;return ans;}
}T;
struct node{
int id,l,r,opt;
inline bool operator <(const node &b)const{
if(r!=b.r)return r<b.r;
return opt<b.opt;
}
};
vector<node>b[N];
vector<node>::iterator it;
inline void add(int u,int v){
e[++tot].n=head[u];e[tot].to=v;head[u]=tot;
}
void getroot(int u,int fa){
dp[u]=0;size[u]=1;
for(int i=head[u];i;i=e[i].n)if(!vis[e[i].to]&&e[i].to!=fa){
int v=e[i].to;
getroot(v,u);
size[u]+=size[v];
dp[u]=max(dp[u],size[v]);
}
dp[u]=max(dp[u],sum-size[u]);
if(dp[u]<dp[root])root=u;
}
void getsize(int u,int fa){
size[u]=1;
for(int i=head[u];i;i=e[i].n)if(!vis[e[i].to]&&e[i].to!=fa){
int v=e[i].to;
getsize(v,u);
size[u]+=size[v];
}
}
void work(int u,int ff,int ma,int mi,int top){
ma=max(ma,u);mi=min(mi,u);
fa[u].push_back(top);
v[u].push_back(mm(mi,ma));
for(int i=head[u];i;i=e[i].n)if(e[i].to!=ff&&!vis[e[i].to]){
int v=e[i].to;
work(v,u,ma,mi,top);
}
}
void getcalc(int u,int fa,int mi,int ma){
mi=min(mi,u);ma=max(ma,u);
b[root].push_back(node{a[u],mi,ma,0});
for(int i=head[u];i;i=e[i].n)if(!vis[e[i].to]&&e[i].to!=fa){
int v=e[i].to;
getcalc(v,u,mi,ma);
}
}
void solve(int u){
vis[u]=1;
getcalc(u,0,u,u);
v[u].push_back(mm(u,u));fa[u].push_back(u);
for(int i=head[u];i;i=e[i].n)if(!vis[e[i].to]){
int v=e[i].to;
root=n+1;sum=size[v];
getroot(v,u);getsize(root,0);
work(v,0,u,u,u);
solve(root);
}
}
int main(){
n=rd();m=rd();
for(int i=1;i<=n;++i)a[i]=rd();
int x,y;
for(int i=1;i<n;++i){
x=rd();y=rd();
add(x,y);add(y,x);
}
root=n+1;sum=n;dp[root]=n+1;
getroot(1,0);getsize(root,0);
solve(root);
int l,r;
for(int i=1;i<=m;++i){
l=rd();r=rd();x=rd();
for(int j=0;j<fa[x].size();++j){
int y=fa[x][j],ls=v[x][j].first,rs=v[x][j].second;
if(ls>=l&&rs<=r){x=y;break;}
}
b[x].push_back(node{i,l,r,1});
}
for(int i=1;i<=n;++i){
sort(b[i].begin(),b[i].end());
for(it=b[i].begin();it!=b[i].end();++it){
if(!it->opt){
if(tong[it->id]<it->l){
if(tong[it->id])T.add(tong[it->id],-1);
T.add(it->l,1);
tong[it->id]=it->l;
}
}
else{
ans[it->id]=T.query(it->l);
}
}
for(it=b[i].begin();it!=b[i].end();++it)
if(!it->opt){
if(tong[it->id])T.add(tong[it->id],-1),tong[it->id]=0;
}
b[i].clear();
}
for(int i=1;i<=m;++i)printf("%d\n",ans[i]);
return 0;
}
[Ynoi2012]D1T3的更多相关文章
- ZROI 暑期高端峰会 A班 Day4 树上数据结构
FBI Warning:本文含有大量人类的本质之一. 你经历过绝望吗? [ZJOI2007]捉迷藏 询问树上最远黑点对. 动态边分治可以比点分治少一个 \(\log\). bzoj3730 咕了. [ ...
- zhengrui集训D1-D5笔记
Day_1 计数 它咕掉了 Day_1 序列数据结构 它咕掉了 Day_2 线性代数 高斯消元\Large{高斯消元}高斯消元 普通版:略 模质数:求逆 模合数:exgcd 逆矩阵\Large{逆矩阵 ...
- [luogu P3953] [noip2017 d1t3] 逛公园
[luogu P3953] [noip2017 d1t3] 逛公园 题目描述 策策同学特别喜欢逛公园.公园可以看成一张$N$个点$M$条边构成的有向图,且没有 自环和重边.其中1号点是公园的入口,$N ...
- 【NOIP2017 D1T3】逛公园
NOIP2017 D1T3 逛公园 题意:给一个有向图,每条边有权值,问从\(1\)到\(N\)的长度不超过最短路长度\(+K\)的路径条数.如果有无数条则输出\(-1\). 思路:我们首先扔掉\(- ...
- [2018HN省队集训D1T3] Or
[2018HN省队集训D1T3] Or 题意 给定 \(n\) 和 \(k\), 求长度为 \(n\) 的满足下列条件的数列的数量模 \(998244353\) 的值: 所有值在 \([1,2^k)\ ...
- noip 2018 D1T3 赛道修建
noip 2018 D1T3 赛道修建 首先考虑二分答案,这时需要的就是对于一个长度求出能在树中选出来的最多的路径条数.考虑到一条路径是由一条向上的路径与一条向下的路径构成,或者仅仅是向上或向下的路径 ...
- [NOIP2018 TG D1T3]赛道修建
题目大意:$NOIP2018\;TG\;D1T3$ 题解:题目要求最短的赛道的长度最大,可以想达到二分答案,接着就是一个显然的树形$DP$. 发现对于一个点,它子树中若有两条链接起来比要求的答案大,一 ...
- 开车旅行 【NOIP2012 D1T3】
开车旅行 [NOIP2012 D1T3] 倍增 首先令\(a[i]\)表示从i出发最近的城市下标,\(b[i]\)表示从i出发第二近的城市下标 可以维护一个\(\text{set<pair< ...
- 嵊州D1T3 睡美人航班
嵊州D1T3 睡美人航班 不知不觉中,我对她的爱意已经达到了 n. 是这样子的,第 1 分钟,我对她的爱意值是 (1, 1). 假如当第 x 分钟时我对她的爱意值是 (a, b),那么第 x + 1 ...
随机推荐
- 前端框架React入门课程【视频】
视频教程列表: http://v1.mukewang.com/1a8228ac-5f7f-48de-b1c5-7d1b8bce9c77/L.mp4 1-1 React入门课程介绍 http://v1. ...
- 【ABAP系列】SAP 读取生产订单 记入文档的货物移动明细
公众号:SAP Technical 本文作者:matinal 原文出处:http://www.cnblogs.com/SAPmatinal/ 原文链接:[ABAP系列]SAP 读取生产订单 记入文档的 ...
- 应用安全 - 安全设备 - WAF原理/检测/绕过
原理 基于Cookie值 Citrix Netscaler(2013年使用广泛) “Citrix Netscaler”会在HTTP返回头部Cookie位置加入“ns_af”的值,可以以此判断为Citr ...
- C语言I作业12——学习总结
1.我学到的内容 二.我的收获 作业 链接 第一次作业 https://www.cnblogs.com/liuxiangjiang/p/11579877.html 第二次作业 https://www. ...
- 老技术记录-C#+SqlServer使用SqlDependency监听数据库表变化
开发环境: .net / C# (.net core理论上也可以) 数据库:MS SQL Server 2005 以上 (我用的sqlserver2012) 功能:SqlDependency提供了一种 ...
- Python学习-第二天-字符串和常用数据结构
Python学习-第二天-字符串和常用数据结构 字符串的基本操作 def main(): str1 = 'hello, world!' # 通过len函数计算字符串的长度 print(len(str1 ...
- P1828 香甜的黄油 (spfa)
[题目描述] 农夫John知道每只奶牛都在各自喜欢的牧场(一个牧场不一定只有一头牛).给出各头牛在的牧场和牧场间的路线,找出使所有牛到达的路程和最短的牧场(他将把糖放在那). [题目链接] https ...
- 利用ssh-copy-id实现SSH无密码登录
第一步: 在远程服务器产生公钥与私钥对: $ ssh-keygen -t rsa 按照提示输入完后,会在~/.ssh目录下生成id_rsa和id_rsa.pub这两个文件 第二步:用ssh-copy- ...
- SpringBoot中设置自定义拦截器
SpringBoot中设置自动以拦截器需要写一个类继承HandlerInterceptorAdapter并重写preHandle方法 例子 public class AuthorityIntercep ...
- Javascript | DOM\DOM树浅析
DOM Document Object Model(文档对象模型) 定义了表示和修改文档所需的方法. DOM对象即为宿主对象,由浏览器厂商定义,用来操作html和xml功能的一类对象的集合.也有人称D ...