CF 375D. Tree and Queries加强版!!!【dfs序分块 大小分类讨论】
题意:
一棵树,询问一个子树内出现次数$\ge k$的颜色有几种,Candy?这个沙茶自带强制在线
吐槽:
本来一道可以离散的莫队我非要强制在线用分块做;上午就开始写了然后发现思路错了...;改 下午继续写....然后发现看大了数据范围卡空间了...;改 然后又发现好多bug...;再改 然后发现TLE了... ;改块的大小....可恶又卡空间了.... ;改short...可恶溢出了;改unsigned short....可恶n总共才1e5怎么练unsigned short也溢出了.....; 开O2...还不行....;然后发现之前把块的大小和数量搞反了....;继续改块的大小再加上有理有据对本题特性的vector优化.....终于A了.................
题解:
一开始想成已经知道k预处理f不用第三维了(md那还用分块干什么)
对出现次数$>S$和$\le S$的分开讨论
预处理$f[i][j][k]$为块i到块j出现次数$[k,S]$的有几种
$s[i][j]$为前i块颜色j出现了几次
询问的时候
两边不完整的块暴力枚举
$>S$的部分不超过$\frac{N}{S}$种,单独暴力枚举(注意如果两边枚举过了就不能重复枚举了)
$[k,S]$的部分直接用预处理的f
#pragma GCC optimize ("O2")
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <cmath>
#include <vector>
using namespace std;
typedef long long ll;
const int N=1e5+, M=, S=;
inline int read(){
char c=getchar();int x=,f=;
while(c<''||c>''){if(c=='-')f=-;c=getchar();}
while(c>=''&&c<=''){x=x*+c-'';c=getchar();}
return x*f;
} int n,Q,col,a[N],u,v,k;
int cou[N], big[N], tot, mark[N];bool biiig[N];
struct edge{int v,ne;}e[N<<];
int cnt,h[N];
inline void ins(int u,int v){
e[++cnt]=(edge){v,h[u]}; h[u]=cnt;
e[++cnt]=(edge){u,h[v]}; h[v]=cnt;
}
int dfc,L[N],R[N];
int t[N];
void dfs(int u,int fa){
L[u]=++dfc; a[dfc]=t[u];
for(int i=h[u];i;i=e[i].ne)
if(e[i].v!=fa) dfs(e[i].v, u);
R[u]=dfc;
} int block,m,pos[N];
struct _blo{int l,r;}b[M];
void ini(){
//block=sqrt(n);
block=;
m=(n-)/block+;
for(int i=;i<=n;i++) pos[i]=(i-)/block+;
for(int i=;i<=m;i++) b[i].l=(i-)*block+, b[i].r=i*block;
b[m].r=n;
} struct Block{
int f[M][M][S], c[N], s[M][N]; void Set0(int x){
for(int i=;i<=col;i++) s[x][i]=s[x-][i];
for(int i=b[x].l; i<=b[x].r; i++) s[x][a[i]]++;
} void Set1(int x){
for(int t=x;t<=m;t++){
for(int i=b[t].l; i<=b[t].r; i++) if(!biiig[ a[i] ]) c[a[i]]++;
for(int i=b[t].l; i<=b[t].r; i++) if(!biiig[ a[i] ] && c[a[i]]>){
int _=s[t-][a[i]] - s[x-][a[i]];
f[x][t][ _+c[a[i]] ]++;
f[x][t][ _ ]--;
c[a[i]]=;
}
for(int i=block; i>=; i--) f[x][t][i]+=f[x][t][i+];
for(int i=; i<=block; i++) f[x][t][i]+=f[x][t-][i];
}
} int Que(int l,int r,int k){
int pl=pos[l], pr=pos[r];
int ans=;
if(pl==pr){
for(int i=l; i<=r; i++) c[a[i]]++;
for(int i=l; i<=r; i++) if(c[a[i]]>) ans+= c[a[i]]>=k, c[a[i]]=;
}else{
for(int i=; i<=tot; i++) mark[ big[i] ]=;
vector<int> v;
int *rr=s[pr], *ll=s[pl-];
for(int i=l; i<=b[pl].r; i++){
mark[ a[i] ]=;
if(rr[a[i]] - ll[a[i]]>=k)
c[a[i]]++, v.push_back(a[i]);
}
for(int i=b[pr].l; i<=r; i++){
mark[ a[i] ]=;
if(rr[a[i]] - ll[a[i]]>=k)
c[a[i]]++, v.push_back(a[i]);
} for(int i=; i<(int)v.size(); i++) if(c[v[i]]>){
int _=s[pr-][v[i]] - s[pl][v[i]];
if(biiig[ v[i] ]) ans+= _+c[v[i]]>=k;
else ans+= (_<k && _+c[v[i]]>=k);
c[v[i]]=;
} if(k<=block) ans+=f[pl+][pr-][k];
for(int i=;i<=tot;i++) if(!mark[ big[i] ])
ans+= s[pr-][big[i]] - s[pl][big[i]] >= k;
}
return ans;
}
}B; int main(){
// freopen("in","r",stdin);
n=read(); Q=read(); ini();
for(int i=;i<=n;i++) a[i]=t[i]=read(), col=max(col, a[i]), cou[a[i]]++;
for(int i=;i<n;i++) ins(read(), read());
dfs(,); for(int i=;i<=col;i++) if(cou[i]>block) big[++tot]=i, biiig[i]=;
for(int i=;i<=m;i++) B.Set0(i);
for(int i=;i<=m;i++) B.Set1(i); while(Q--){
u=read(); k=read();
printf("%d\n", B.Que(L[u], R[u], k) );
}
}
CF 375D. Tree and Queries加强版!!!【dfs序分块 大小分类讨论】的更多相关文章
- Codeforces 375D Tree and Queries(DFS序+莫队+树状数组)
题目链接 Tree and Queries 题目大意 给出一棵树和每个节点的颜色.每次询问$vj, kj$ 你需要回答在以$vj$为根的子树中满足条件的的颜色数目, 条件:具有该颜色的节点数量至少 ...
- CF 375D. Tree and Queries【莫队 | dsu on tree】
题意: 一棵树,询问一个子树内出现次数$≥k$的颜色有几种 强制在线见上一道 用莫队不知道比分块高到哪里去了,超好写不用调7倍速度!!! 可以用分块维护出现次数这个权值,实现$O(1)-O(\sqrt ...
- CodeForces 375D Tree and Queries 莫队||DFS序
Tree and Queries 题意:有一颗以1号节点为根的树,每一个节点有一个自己的颜色,求出节点v的子数上颜色出现次数>=k的颜色种类. 题解:使用莫队处理这个问题,将树转变成DFS序区间 ...
- CodeForces - 375D Tree and Queries (莫队+dfs序+树状数组)
You have a rooted tree consisting of n vertices. Each vertex of the tree has some color. We will ass ...
- Codeforces 375D - Tree and Queries(dfs序+莫队)
题目链接:http://codeforces.com/contest/351/problem/D 题目大意:n个数,col[i]对应第i个数的颜色,并给你他们之间的树形关系(以1为根),有m次询问,每 ...
- 【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 ...
- SPOJ Query on a tree III (树剖(dfs序)+主席树 || Splay等平衡树)(询问点)
You are given a node-labeled rooted tree with n nodes. Define the query (x, k): Find the node whose ...
- POJ 3321 Apple Tree (树状数组+dfs序)
题目链接:http://poj.org/problem?id=3321 给你n个点,n-1条边,1为根节点.给你m条操作,C操作是将x点变反(1变0,0变1),Q操作是询问x节点以及它子树的值之和.初 ...
- SP1487 PT07J - Query on a tree III 主席树+dfs序
Code: #include<iostream> #include<cstdio> #include<algorithm> #include<string&g ...
随机推荐
- I Hate It(线段树点修改区间查询)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1754 I Hate It Time Limit: 9000/3000 MS (Java/Others) ...
- Xshell无法连接到LINUX虚拟机
首先与遇到的情况是,在虚拟机下安装了Linux后,xshell无法连接远程的虚拟机. 我遇到的情况是虚拟机可以ping 主机,主机确ping不了虚拟机. 使用的VM设置了两个网卡,一个nat 一个h ...
- Flume环境搭建_五种案例
Flume环境搭建_五种案例 http://flume.apache.org/FlumeUserGuide.html A simple example Here, we give an example ...
- tp路由+伪静态+去掉index.php
浏览:10536 发布日期:2013/10/08 分类:技术分享 关键字: 路由 伪静态 去掉index.php 之前一个网友说能不能达到这样的效果,www.olcms.com/news/id.htm ...
- ftp上传文件,本地安装了,服务器上也需要在也安装一个ftp
服务器需要配置FTP服务: 你说的在你自己电脑上安装的只是一个FTP软件,用于连接远程服务器进行上传和下载文件的. 追问 在本地已经安装了,链接的话要在服务器上也安装一个吗? 追答 额,你有FTP服务 ...
- // 关闭调试模式 define('APP_DEBUG', false);
调试模式的优势在于: 开启日志记录,任何错误信息和调试信息都会详细记录,便于调试: 关闭模板缓存,模板修改可以即时生效: 记录SQL日志,方便分析SQL: 关闭字段缓存,数据表字段修改不受缓存影响: ...
- Flexible Box布局基础知识详解
1.基本概念,借用阮一峰老师的一张图: 容器默认存在两根轴:水平的主轴(main axis)和垂直的交叉轴(cross axis).主轴的开始位置(与边框的交叉点)叫做main start,结束位置叫 ...
- .net 和 core2.0 数据库连接字符串
Asp.net Core 数据库离线文件的连接(引自“张不水”兄的研究成果.) 一.绝对路径: "DefaultConnection": "Data Source=(lo ...
- 有关linux下redis overcommit_memory的问题
公司的几台Redis服务器出现不明故障,查看Redis日志,发现如下提示: 1 [34145] 01 Jan 17:42:02 # WARNING overcommit_memory is set t ...
- linux运维学习
export 和unset 设置和取消变量 echo 的双引号和单引号的区别:双引号里的会被替换,单引号里的都会直接输出.