Problem Distance in tree

题目大意

给出一棵树,求这棵树上有多少个最短距离为k的点对。

Solution

这个题目可以用点分治来做,然而我到现在还是没有学会点分治,所以只好用树形dp了。

这个题目,我们可以将其转化为一个个子树中搞事情,再慢慢合并。

设f[i][j]为以i为根的子树中距离根距离为j的点有多少个。

对于一个点u,我们枚举它的子节点v,则我们可以计算出经过u-v这条边的答案

我们枚举j=1->k,则ans+=f[u][j]*f[v][k-j-1];

枚举完以后,我们将这一棵子树合并到u节点去,以便统计u的其它子树。

具体过程我们只需要dfs一遍,对于每个叶子结点x,f[x][0]=1。

然后不断向上回溯,边回溯边进行如上计算,最后即可算出正确答案。

具体可以参考一下代码。

AC Code

 #include <iostream>
#include <cstdio>
#include <cstring>
using namespace std;
struct node{
int to,next;
}e[];
int n,k,u,v,ans=,tot=;
int h[],f[][];
void add(int u,int v){
e[++tot].to=u;e[tot].next=h[v];h[v]=tot;
e[++tot].to=v;e[tot].next=h[u];h[u]=tot;
}
void dfs(int x,int last){
f[x][]=;
for(int i=h[x];~i;i=e[i].next){
if(e[i].to!=last){
dfs(e[i].to,x);
for(int j=;j<k;j++)ans+=f[x][j]*f[e[i].to][k-j-];
for(int j=;j<=k;j++)f[x][j]+=f[e[i].to][j-];
}
}
}
int main(){
// freopen("cf161d.in","r",stdin);
memset(h,-,sizeof(h));
scanf("%d%d",&n,&k);
for(int i=;i<n;i++){
scanf("%d%d",&u,&v);
add(u,v);
}
dfs(,);
printf("%d",ans);
}
 1 #include <iostream>
2 #include <cstdio>
3 #include <cstring>
4 using namespace std;
5 struct node{
6 int next,to;
7 }e[200010];
8 bool neko[100010];
9 int n,m,d,x,u,v,path[100010],h[100010],ans=0,tot=0;
10 int pathp[100010],pathn[100010];
11 void add(int u,int v){
12 e[++tot].to=u;e[tot].next=h[v];h[v]=tot;
13 e[++tot].to=v;e[tot].next=h[u];h[u]=tot;
14 }
15 void dfspath(int x,int last){
16 path[x]=-2333333;pathp[x]=-2333333;
17 for(int i=h[x];~i;i=e[i].next){
18 if(e[i].to!=last){
19 dfspath(e[i].to,x);
20 if(path[x]<path[e[i].to]+1){
21 pathp[x]=path[x];
22 path[x]=path[e[i].to]+1;
23 pathn[x]=e[i].to;
24 }else if(pathp[x]<path[e[i].to]+1)pathp[x]=path[e[i].to]+1;
25 }
26 }
27 if(neko[x]&&path[x]<-2000000)path[x]=0;
28 if(neko[x]&&pathp[x]<-2000000)pathp[x]=0;
29 }
30 void dpdfs(int x,int last){
31 if(~last)
32 if(pathn[last]==x){
33 if(pathp[last]+1>path[x]){
34 pathp[x]=path[x];
35 path[x]=pathp[last]+1;
36 pathn[x]=0;
37 }else if(pathp[last]+1>pathp[x])pathp[x]=pathp[last]+1;
38 }else{
39 if(path[last]+1>path[x]){
40 pathp[x]=path[x];
41 path[x]=path[last]+1;
42 pathn[x]=0;
43 }else if(path[last]+1>pathp[x])
44 pathp[x]=path[last]+1;
45 }
46 if(path[x]<=d)ans++;
47 for(int i=h[x];~i;i=e[i].next)
48 if(e[i].to!=last)dpdfs(e[i].to,x);
49 }
50 int main(){
51 // freopen("cf337d.in","r",stdin);
52 memset(h,-1,sizeof(h));
53 scanf("%d%d%d",&n,&m,&d);
54 for(int i=1;i<=m;i++)
55 scanf("%d",&x),neko[x]=1;
56 for(int i=1;i<n;i++){
57 scanf("%d%d",&u,&v);
58 add(u,v);
59 }
60 dfspath(1,1);
61 dpdfs(1,-1);
62 printf("%d",ans);
63 }
 
 
标签: 树形dp
好文要顶 关注我 收藏该文  
0
0
 
 
 
posted @ 2017-07-09 16:09 skylynf 阅读(4) 评论(1) 编辑 收藏
 
评论列表
 

#1楼 2017-07-09 20:02 ywwyww 

 

公告

昵称:skylynf
园龄:5天
粉丝:0
关注:0

 
< 2017年7月 >
25 26 27 28 29 30 1
2 3 4 5 6 7 8
9 10 11 12 13 14 15
16 17 18 19 20 21 22
23 24 25 26 27 28 29
30 31 1 2 3 4 5

[CF161D]Distance in Tree-树状dp的更多相关文章

  1. poj2486--Apple Tree(树状dp)

    Apple Tree Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 7789   Accepted: 2606 Descri ...

  2. Codeforces 161D Distance in Tree(树型DP)

    题目链接 Distance in Tree $k <= 500$ 这个条件十分重要. 设$f[i][j]$为以$i$为子树,所有后代中相对深度为$j$的结点个数. 状态转移的时候,一个结点的信息 ...

  3. POJ 2486 Apple Tree [树状DP]

    题目:一棵树,每个结点上都有一些苹果,且相邻两个结点间的距离为1.一个人从根节点(编号为1)开始走,一共可以走k步,问最多可以吃多少苹果. 思路:这里给出数组的定义: dp[0][x][j] 为从结点 ...

  4. CodeForces 160D - Distance in Tree 树型DP

    题目给了512MB的空间....用dp[k][i]代表以k为起点...往下面走(走直的不打岔)i步能有多少方案....在更新dp[k][i]过程中同时统计答案.. Program: #include& ...

  5. Codeforces 461B - Appleman and Tree 树状DP

    一棵树上有K个黑色节点,剩余节点都为白色,将其划分成K个子树,使得每棵树上都仅仅有1个黑色节点,共同拥有多少种划分方案. 个人感觉这题比較难. 如果dp(i,0..1)代表的是以i为根节点的子树种有0 ...

  6. D. Distance in Tree(树型Dp计数)

    \(其实思路都能想到一点,就是去重这里特别麻烦,没有好的思路.\) \(设dp[i][j]为以i为根深度为j的节点数量\) \(dp[parent][j]=\sum{dp[son][j-1]}\) \ ...

  7. HDU 4714 Tree2cycle(树状DP)(2013 ACM/ICPC Asia Regional Online ―― Warmup)

    Description A tree with N nodes and N-1 edges is given. To connect or disconnect one edge, we need 1 ...

  8. 树状DP (poj 2342)

    题目:Anniversary party 题意:给出N各节点的快乐指数,以及父子关系,求最大快乐指数和(没人职员愿意跟直接上司一起玩): 思路:从底向上的树状DP: 第一种情况:第i个员工不参与,F[ ...

  9. poj3659树状DP

    Cell Phone Network Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 6273   Accepted: 225 ...

  10. hdu 1561 The more, The Better_树状dp

    题目链接 题意:给你一棵树,各个节点都有价值(除根节点),从根节点出发,选择m个节点,问最多的价值是多小. 思路:很明显是树状dp,遍历树时背包最优价值,dp[i][k]=max{dp[i][r]+d ...

随机推荐

  1. int 与 int *

    #include <iostream>using namespace std;int QKPass(int* , int , int);  //若声明为 int QKPass(int, i ...

  2. msseces.exe频繁出错的原因和解决方法?

    关机时会报错,什么内存为只读.. 以下是官方给的解决方案,相信对大部分用户都起作用,在这分享给大家. 对于当前遇到的问题,有可能是由于程序冲突导致.因此建议进入干净启动状态再确认问题是否发生: 1.如 ...

  3. Java中设计模式之单例设计模式-1

    单例作用 1 节省内存 2 可以避免多种状态导致状态冲突 单例的创建步骤 1 私有化构造方法 2 私有化声明的属性 3 getInstance 4 方法需要静态 单例分类 1.懒汉式 2.饿汉式 两种 ...

  4. C++抽象编程·运算符重载与友元函数

    运算符重载(Operator overloading) 从我们在几个前篇的类的层次介绍中可以知道,C++可以扩展标准运算符,使其适用于新类型.这种技术称为运算符重载. 例如,字符串类重载+运算符,使其 ...

  5. if 一元二次方程求根

    if 语句 - 只有当指定条件为 true 时,使用该语句来执行代码 if...else 语句 - 当条件为 true 时执行代码,当条件为 false 时执行其他代码 if...else if... ...

  6. IE6不支持position:fixed的解决方法

    解决IE6不支持position:fixed的方法,非常简单,具体调用请参考下面: /*让position:fixed在IE6下可用! */ .fixed-top /* 头部固定 */{positio ...

  7. 解决kubuntu(KDE4.8.5桌面环境)找不到中文语言包

    原始日期:2013-12-30 23:16 接触linux的想必都知道KDE平台,kde精美的界面是其一大特色,不过美中不足的是,很多新手在安装完KDE后,界面包括菜单选项等都是英文界面,对于英语水平 ...

  8. node.js 发送http 请求

    自己研究了一下 node.js 的 http模块  下面为想服务器发送请求的代码 ,通过学习了解http 请求的过程 ,node.js 对http请求的原始封装比较低,以前php 可以用$_GET , ...

  9. promise(3) '静态'方法

    要是人没有梦想,跟咸鱼又有什么两样了?一直恐惧读源码,哪怕是一个简单的库也是读百来行遇到难点就放弃了.对于新的东西也仅仅是知道它拿来干什么,社区资源在哪里,要用时就突击文档资源使用即可.未有过深入之心 ...

  10. jQuery.merge( first, second )返回: Array

    jQuery.merge( first, second )返回: Array描述: 合并两个数组内容到第一个数组.first类型: Array第一个用于合并的数组,其中将会包含合并后的第二个数组的内容 ...