题目链接: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. iOS 中的 block 是如何持有对象的

    Block 是 Objective-C 中笔者最喜欢的特性,它为 Objective-C 这门语言提供了强大的函数式编程能力,而最近苹果推出的很多新的 API 都已经开始原生的支持 block 语法, ...

  2. css实现响应式全屏背景

    利用css中 background-size:cover  填充整个viewport 注意: 一张背景图像素5000px*5000px在pc端 缩放都基本满足要求 不会出现模糊失真: 但是在移动端使用 ...

  3. jQuery的自定义事件——滚轮

    这个案例类似于在地图上滚动滚轮,能缩小或者放大地图,分别用zoomIn和zoomOut来命名. 代码如下: //JS部分:<script src="jquery-1.10.2.min. ...

  4. 【字符串排序,技巧!】UVa 10905 - Children’s Game

    There are lots of number games for children. These games are pretty easy to play but not so easy to ...

  5. 聊聊css盒子模型

    css盒子模型原理: 在网页设计中常听的属性名:内容(content).填充/内边距(padding).边框(border).外边距(margin), CSS盒子模式都具备这些属性. 这些属性我们可以 ...

  6. 【C#4.0图解教程】笔记(第1章~第8章)

    第1章 C#和.NET框架 1..NET框架的组成 .NET框架由三部分组成(严格来说只有CLR和FCL(框架类库)两部分),如图 执行环境称为:CLR(公共语言运行库),它在运行期管理程序的执行. ...

  7. String类中toCharArray()方法的用法

    该方法的作用是返回一个字符数组,该字符数组中存放了当前字符串中的所有字符 eg:  public class class6_3 { public static void main(String arg ...

  8. c#转义字符串中的所有正则特殊字符

    /// <summary> /// 转义字符串中所有正则特殊字符 /// </summary> /// <param name="input"> ...

  9. IOS 高级开发 KVC(二)

    前一篇博客最后介绍了KVC 再json 转模型时遇到一些问题.今天接着来介绍KVC 的其他用法.其实我们在一开始的时候就一直再强调命名的重要性.命名规范是KVC 存活的基础.如果没有这个条件支撑,那么 ...

  10. unity发布ios游戏总结

    自己做了几个ios的小游戏,因此总结了一点经验 判断按钮要用unity里面的button不要用OnMouseDown()之类的函数,否则拒绝原因为缺少ios特征 排行榜之类的本地存储数据,不要用本地本 ...