题目

给你一棵 \(n\) 个节点的树,每个节点有一种颜色,有 \(m\) 次查询操作。

查询操作给定参数 \(l\) \(r\) \(x\),需输出:

将树中编号在 \([l,r]\) 内的所有节点保留,\(x\) 所在连通块中颜色种类数。

每次查询操作独立。


分析

首先点分治能够通过经过当前重心的连通块表示出所有的连通块

考虑一个点 \(y\) 处在 \(x\) 的连通块中

当且仅当 \(x\) 到 \(y\) 的路径上的点下标在 \([l,r]\) 范围内。

那么每次分治到一个新的重心,求出每个点到根节点的最小下标 \(L\) 和最大下标 \(R\)。

可以发现这个询问可以在这个重心被计算当且仅当 \(l\leq L\leq R\leq r\),并且只要询问过就不需要递归下去。

那么只要这个条件满足就把它加入待询问的数组中,相当于在该重心的子树中提供 \((L,R,Col)\),询问颜色个数。

那么将询问和这些点均按 \(l\) 从大到小排序,然后对于颜色记录前驱(HH的项链),

尽量使其出现的右端点尽量靠左(取最小值),然后用树状数组维护不超过其右端点的右端点个数。

时间复杂度 \(O((n+Q)\log ^2n)\)


代码

#include <cstdio>
#include <cctype>
#include <cstring>
#include <algorithm>
using namespace std;
const int N=100011; struct node{int y,next;}e[N<<1]; struct rec{int l,r,z;}E[N],b[N];
int siz[N],big[N],as[N],hs[N],rk[N],n,et=1,Q,ans[N],v[N],root,tot,TOT,a[N],las[N],c[N];
int iut(){
int ans=0; char c=getchar();
while (!isdigit(c)) c=getchar();
while (isdigit(c)) ans=ans*10+c-48,c=getchar();
return ans;
}
void print(int ans){
if (ans>9) print(ans/10);
putchar(ans%10+48);
}
void update(int x,int y){
for (;x<=n;x+=-x&x) c[x]+=y;
}
int query(int x){
int ans=0;
for (;x;x-=-x&x) ans+=c[x];
return ans;
}
int max(int a,int b){return a>b?a:b;}
int min(int a,int b){return a<b?a:b;}
void dfs(int x,int fa){
siz[x]=1,big[x]=0;
for (int i=as[x];i;i=e[i].next)
if (!v[e[i].y]&&e[i].y!=fa){
dfs(e[i].y,x),siz[x]+=siz[e[i].y];
big[x]=max(big[x],siz[e[i].y]);
}
big[x]=max(big[x],big[0]-siz[x]);
if (big[x]<=big[root]) root=x;
}
void Dfs(int x,int fa,int l,int r){
b[++tot]=(rec){l,r,a[x]};
for (int i=hs[x],now=0;i;i=E[i].z)
if (E[i].l<=l&&r<=E[i].r){
rk[++TOT]=i;
if (!now) hs[x]=E[i].z;
else E[now].z=E[i].z;
}else now=i;
for (int i=as[x];i;i=e[i].next)
if (!v[e[i].y]&&e[i].y!=fa)
Dfs(e[i].y,x,min(l,e[i].y),max(r,e[i].y));
}
bool cmp0(rec x,rec y){return x.l>y.l;}
bool cmp1(int x,int y){return E[x].l>E[y].l;}
void Dp(int x){
v[x]=1,tot=TOT=0,Dfs(x,0,x,x);
sort(b+1,b+1+tot,cmp0),sort(rk+1,rk+1+TOT,cmp1);
int j=1;
for (int i=1;i<=TOT;++i){
for (;j<=tot&&b[j].l>=E[rk[i]].l;++j)
if (las[b[j].z]>b[j].r){
update(las[b[j].z],-1);
update(las[b[j].z]=b[j].r,1);
}
ans[rk[i]]=query(E[rk[i]].r);
}
for (int i=1;i<j;++i)
if (las[b[i].z]!=0x3f3f3f3f)
update(las[b[i].z],-1),las[b[i].z]=0x3f3f3f3f;
for (int i=as[x];i;i=e[i].next) if (!v[e[i].y])
big[0]=siz[e[i].y],dfs(e[i].y,root=0),Dp(root);
}
int main(){
n=iut(),Q=iut();
memset(las,0x3f,sizeof(las));
for (int i=1;i<=n;++i) a[i]=iut();
for (int i=1;i<n;++i){
int x=iut(),y=iut();
e[++et]=(node){y,as[x]},as[x]=et;
e[++et]=(node){x,as[y]},as[y]=et;
}
for (int i=1;i<=Q;++i){
int l=iut(),r=iut(),x=iut();
E[i]=(rec){l,r,hs[x]},hs[x]=i;
}
big[0]=n,dfs(1,0),Dp(root);
for (int i=1;i<=Q;++i) print(ans[i]),putchar(10);
return 0;
}

#点分治,树状数组#洛谷 5311 [Ynoi2011] 成都七中的更多相关文章

  1. 树状数组 洛谷P3616 富金森林公园

    P3616 富金森林公园 题目描述 博艾的富金森林公园里有一个长长的富金山脉,山脉是由一块块巨石并列构成的,编号从1到N.每一个巨石有一个海拔高度.而这个山脉又在一个盆地中,盆地里可能会积水,积水也有 ...

  2. BZOJ_3262_陌上花开_CDQ分治+树状数组

    BZOJ_3262_陌上花开_CDQ分治+树状数组 Description 有n朵花,每朵花有三个属性:花形(s).颜色(c).气味(m),用三个整数表示. 现在要对每朵花评级,一朵花的级别是它拥有的 ...

  3. 【BZOJ4553】[Tjoi2016&Heoi2016]序列 cdq分治+树状数组

    [BZOJ4553][Tjoi2016&Heoi2016]序列 Description 佳媛姐姐过生日的时候,她的小伙伴从某宝上买了一个有趣的玩具送给他.玩具上有一个数列,数列中某些项的值可能 ...

  4. BZOJ 1176 Mokia CDQ分治+树状数组

    1176: [Balkan2007]Mokia Time Limit: 30 Sec  Memory Limit: 162 MBSubmit: 1854  Solved: 821[Submit][St ...

  5. 【bzoj3262】陌上花开 CDQ分治+树状数组

    题目描述 有n朵花,每朵花有三个属性:花形(s).颜色(c).气味(m),又三个整数表示.现要对每朵花评级,一朵花的级别是它拥有的美丽能超过的花的数量.定义一朵花A比另一朵花B要美丽,当且仅当Sa&g ...

  6. 【bzoj2225】[Spoj 2371]Another Longest Increasing CDQ分治+树状数组

    题目描述 给定N个数对(xi, yi),求最长上升子序列的长度.上升序列定义为{(xi, yi)}满足对i<j有xi<xj且yi<yj. 样例输入 8 1 3 3 2 1 1 4 5 ...

  7. BZOJ_2253_[2010 Beijing wc]纸箱堆叠 _CDQ分治+树状数组

    BZOJ_2253_[2010 Beijing wc]纸箱堆叠 _CDQ分治+树状数组 Description P 工厂是一个生产纸箱的工厂.纸箱生产线在人工输入三个参数 n p a , , 之后, ...

  8. BZOJ_3295_[Cqoi2011]动态逆序对_CDQ分治+树状数组

    BZOJ_3295_[Cqoi2011]动态逆序对_CDQ分治+树状数组 Description 对于序列A,它的逆序对数定义为满足i<j,且Ai>Aj的数对(i,j)的个数.给1到n的一 ...

  9. BZOJ_2225_[Spoj 2371]Another Longest Increasing_CDQ 分治+树状数组

    BZOJ_2225_[Spoj 2371]Another Longest Increasing_CDQ 分治+树状数组 Description        给定N个数对(xi, yi),求最长上升子 ...

  10. BZOJ_2683_简单题&&BZOJ_1176_[Balkan2007]Mokia_CDQ分治+树状数组

    BZOJ_2683_简单题&&BZOJ_1176_[Balkan2007]Mokia_CDQ分治+树状数组 Description 维护一个W*W的矩阵,初始值均为S.每次操作可以增加 ...

随机推荐

  1. duilib 入坑

    记录 duilib 开发遇到的问题 当前最新的 duilib 版本更新是在  2019-4-28-2,从 vcpkg 查询得知 我的机器是 windows 10,vs2019 我是从 duilib 库 ...

  2. MyBatisSystemException: nested exception is org.apache.ibatis.exceptions.TooManyResultsException: Expected one result (or null) to be returned by selectOne(), but found: 2

    描述: 2021-08-xx 13:31:30.049 DEBUG 9208 : ==> Preparing: SELECT SUM(end_vt) - SUM(start_vt) FROM s ...

  3. Fpga开发笔记(二):高云FPGA发开发软件Gowin和高云fpga基本开发过程

    前言   本篇安装高云的开发软件Gowin,并且描述了一个基于高云fpga的程序的开发环境和完整的下载运行过程.   Gowin软件 概述   Gowin 软件是广东高云半导体股份有限公司的 FPGA ...

  4. OsgEarth开发笔记(二):Osg3.6.3+OsgEarth3.1+vs2019x64开发环境搭建(中)

    上一篇:<OsgEarth开发笔记(一):Osg3.6.3+OsgEarth3.1+vs2019x64开发环境搭建(上)>下一篇:敬请期待-   前言  上一篇编译了osg和osgCurl ...

  5. python模块imghdr-----推测图像类型

    官方文档 https://docs.python.org/zh-cn/3/library/imghdr.html#module-imghdr 用处 模块推测文件或字节流中的图像的类型 imghdr.w ...

  6. 【Filament】绘制圆形

    1 前言 ​ Filament环境搭建中介绍了 Filament 的 Windows 和 Android 环境搭,绘制三角形中介绍了绘制纯色和彩色三角形,绘制矩形中介绍了绘制纯色和彩色矩形,本文将使用 ...

  7. 第130篇:BOM(window对象)

    好家伙,本篇为<JS高级程序设计>第十二章"BOM"学习笔记   什么是BOM? BOM(Browser Object Model) 是指浏览器对象模型,是用于描述这种 ...

  8. 【Azure Redis 缓存】使用StackExchange.Redis,偶发ERROR - Timeout performing HSET (15000ms)

    问题描述 使用StackExchange.Redis 作为Redis客户端SDK,连接Azure Redis服务,长期运行后发现,每天都偶发 Timeout Error. 错误消息如下: StackE ...

  9. 【Azure 应用服务】在Azure Funciton中使用Powershell脚本函数,需要存储一些变量值如何解决?

    问题描述 使用Azure Function创建Powershell脚本来执行函数,在使用中需要存储一些不重要的参数.一般情况,存储的问题都是交给DB,Storage等来解决.但是有没有一种简单的办法呢 ...

  10. 【Azure Redis 缓存】如何使得Azure Redis可以仅从内网访问? Config 及 Timeout参数配置

    问题描述 问题一:Redis服务,如何可以做到仅允许特定的子网内的服务器进行访问? 问题二:Redis服务,timeout和keepalive的设置是怎样的?是否可以配置成timeout 0? 问题三 ...