题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4358

题意:以1为根节点含有N(N <= 1e5)个结点的树,每个节点有一个权值(weight <= 1e9)。之后有m(m <= 1e5)次查询,每次查询以节点u为子树的树中,权值出现k次的权值有多少个?

Sample Input
1
3 1 (n,k)
1 2 2
1 2
1 3
3 (m)
2 1 3
 
Sample Output
Case #1:
1
1
1
 
思路:建好树之后,dfs得到L[x],R[x].即将树形结构变成了线性的结构。并且由于问的是权值出现k次的权值有多少个,和权值的大小无关。就可以离散化,在dfs中把节点重新排序,并且重新赋值.(我们只关心dfs序之后的id和val);这样之后直接跑莫队即可;
细节:至于离散化,直接使用了vector和lower_bound()来合并相同的值;之后输入时建好查询的q[]即可.
#pragma comment(linker, "/STACK:1024000000")
#include<bits/stdc++.h>
using namespace std;
#define rep0(i,l,r) for(int i = (l);i < (r);i++)
#define rep1(i,l,r) for(int i = (l);i <= (r);i++)
#define rep_0(i,r,l) for(int i = (r);i > (l);i--)
#define rep_1(i,r,l) for(int i = (r);i >= (l);i--)
#define MS0(a) memset(a,0,sizeof(a))
#define MS1(a) memset(a,-1,sizeof(a))
#define MSi(a) memset(a,0x3f,sizeof(a))
#define inf 0x3f3f3f3f
#define lson l, m, rt << 1
#define rson m+1, r, rt << 1|1
typedef pair<int,int> PII;
#define A first
#define B second
#define MK make_pair
#define pb push_back
typedef __int64 ll;
template<typename T>
void read1(T &m)
{
T x=,f=;char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=-;ch=getchar();}
while(ch>=''&&ch<=''){x=x*+ch-'';ch=getchar();}
m = x*f;
}
template<typename T>
void read2(T &a,T &b){read1(a);read1(b);}
template<typename T>
void read3(T &a,T &b,T &c){read1(a);read1(b);read1(c);}
template<typename T>
void out(T a)
{
if(a>) out(a/);
putchar(a%+'');
}
const int M = ;
int head[M],tot,a[M],dfs_clock;
int T,kase = ,i,j,k,n,m,top,cnt[M];
struct Edge{
int to,w,Next;
Edge(){}
Edge(int to,int w,int Next):to(to),w(w),Next(Next){}
}e[M<<];
inline void ins(int u,int v,int w = )
{
e[++tot] = Edge{v,w,head[u]};
head[u] = tot;
}
vector<int> p;
void init()
{
dfs_clock = tot = ;
MS0(head);p.clear();
MS0(cnt);
}
int L[M],R[M],val[M],ans[M];
void dfs(int u,int pre)
{
L[u] = ++dfs_clock;
val[dfs_clock] = a[u];//关系的只是dfs序之后的序号;
for(int d = head[u];d;d = e[d].Next){
int v = e[d].to;
if(v == pre) continue;
dfs(v,u);
}
R[u] = dfs_clock;
}
struct data{
int l,r,id,block;
data(){}
data(int l,int r,int id,int block):l(l),r(r),id(id),block(block){}
}q[M];
bool cmp(const data& a,const data& b)
{
return a.block - b.block?a.block < b.block:a.r < b.r;
}
void update(int pos,int add)
{
if(cnt[val[pos]] == k) top--;
else if(cnt[val[pos]] + add == k) top++;
cnt[val[pos]] += add;
}
void solve()
{
top = ;
for(int i = ,l = ,r = ;i <= m;i++){
while(r < q[i].r) update(++r,);
while(r > q[i].r) update(r--,-);
while(l < q[i].l) update(l++,-);
while(l > q[i].l) update(--l,);
ans[q[i].id] = top;
}
}
int main()
{
read1(T);
while(T--){
init();
read2(n,k);
rep1(i,,n) read1(a[i]),p.pb(a[i]);
sort(p.begin(),p.end());
rep1(i,,n) //离散化;
a[i] = lower_bound(p.begin(),p.end(),a[i]) - p.begin();
int u,v;
rep0(i,,n){
read2(u,v);
ins(u,v);ins(v,u);
}
dfs(,-);
int block = sqrt(n);
read1(m);
rep1(i,,m){
read1(u);
q[i] = data(L[u],R[u],i,L[u]/block);
}
sort(q+,q++m,cmp);
solve();
if(kase) puts("");
printf("Case #%d:\n",++kase);
rep1(i,,m){
out(ans[i]);
puts("");
}
}
return ;
}

hdu 4358 Boring counting 离散化+dfs序+莫队算法的更多相关文章

  1. HDU 4358 Boring counting dfs序+莫队算法

    题意:N个节点的有根树,每个节点有一个weight.有Q个查询,问在以u为根的子树中,有恰好出现了K次的weight有多少种. 这是第一次写莫队算法,之前也只是偶有耳闻. 看了别人的代码打的,还是贴上 ...

  2. HDU 4358 Boring counting(莫队+DFS序+离散化)

    Boring counting Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 98304/98304 K (Java/Others) ...

  3. hdu 4358 Boring counting dfs序+莫队+离散化

    Boring counting Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 98304/98304 K (Java/Others) ...

  4. HDU - 4358 Boring counting (dsu on tree)

    Boring counting: http://acm.hdu.edu.cn/showproblem.php?pid=4358 题意: 求一棵树上,每个节点的子节点中,同一颜色出现k次 的 个数. 思 ...

  5. HDU - 4358 Boring counting (树上启发式合并/线段树合并)

    题目链接 题意:统计树上每个结点中恰好出现了k次的颜色数. dsu on tree/线段树合并裸题. 启发式合并1:(748ms) #include<bits/stdc++.h> usin ...

  6. Codeforces 375D - Tree and Queries(dfs序+莫队)

    题目链接:http://codeforces.com/contest/351/problem/D 题目大意:n个数,col[i]对应第i个数的颜色,并给你他们之间的树形关系(以1为根),有m次询问,每 ...

  7. Codeforces 375D Tree and Queries(DFS序+莫队+树状数组)

    题目链接  Tree and Queries 题目大意  给出一棵树和每个节点的颜色.每次询问$vj, kj$ 你需要回答在以$vj$为根的子树中满足条件的的颜色数目, 条件:具有该颜色的节点数量至少 ...

  8. CF600E Lomsat gelral (dfs序+莫队)

    题面 题解 看到网上写了很多DSU和线段树合并的题解,笔者第一次做也是用的线段树合并,但在原题赛的时候却怕线段树合并调不出来,于是就用了更好想更好调的莫队. 这里笔者就说说莫队怎么做吧. 我们可以通过 ...

  9. HDU 4358 Boring counting 树状数组+思路

    研究了整整一天orz……直接上官方题解神思路 #include <cstdio> #include <cstring> #include <cstdlib> #in ...

随机推荐

  1. Validation

    Validation A simple but powerful Validation Engine, in a Laravel-esque style. Its Validation Rules a ...

  2. (转载)c语言指针学习

    前言 近期俄罗斯的陨石.四月的血月.五月北京的飞雪以及天朝各种血腥和混乱,给人一种不详的预感.佛祖说的末法时期,五浊恶世 ,十恶之世,人再无心法约束,道德沦丧,和现在正好吻合.尤其是在天朝,空气,水, ...

  3. iOS开发之静态库.a的制作教程

    第一种方法:直接新建一个工程. 1.新建项目-> 选择 “Cocoa Touch Static Library” 2.添加库需要包含的源代码,将你工程里的代码添加到打静态库工程里: 3.配置一下 ...

  4. Debug 之 VS2010网站生成成功,但是发布失败

    用vs做好了网站.清理解决方案和重新生成解决方案都可以.但是发布不能成功.发布不能成功,有错误还好,郁闷的是竟然没有错误提示. 解决方法: 1.发布文件夹权限问题.重新找个地方建立一个发布文件夹即可. ...

  5. IPC with pipes, also dup2 redirect stream handle

    #include <stdio.h> #include <sys/types.h> #include <sys/wait.h> #include <unist ...

  6. Number of failed login attempts exceeds threshold value

    OEM发出好多告警,Number of failed login attempts exceeds threshold value. profile当前配置10次失败就会锁定user 查看下,dba_ ...

  7. [转]c#.NET和VB.NET语法的比较

    本文转自:http://www.cnblogs.com/lify0407/archive/2007/08/01/838589.html c#.NET和VB.NET语法的比较   VB.NET C# C ...

  8. Chrome&FF&Opera&下DIV不设置高度显示背景颜色和边框的办法

    今天在排版的时候,外层的div不写高度的话背景颜色和边框没法办法显示,但是在IE下面就可以,这个有三个解决办法. 第一: 直接给最外层的div设置高度(不推荐). 第二: 在内部每个div后添加一个清 ...

  9. 【C语言】5-结构体

    一.什么是结构体 * 在实际应用中,我们通常需要由不同类型的数据来构成一个整体,比如学生这个整体可以由姓名.年龄.身高等数据构成,这些数据都具有不同的类型,姓名可以是字符串类型,年龄可以是整型,身高可 ...

  10. webstorm添加vue模板支持

    字谕纪泽: 八月一日,刘曾撰来营,接尔第二号信并薛晓帆信,得悉家中四宅平定,至以为尉. 汝读”四书”无甚心得,由不能虚心涵泳,切己体察.朱子教人读书之法,此二语最为精当.尔现读”离娄”,即如“离娄”首 ...