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. NetFramework各个版本的特性笔记

    公式记忆: .Net 2.0=CLR+BCL+C#(VB.Net)+Win Form+Web Form .Net 3.0=.Net 2.0+WCF+WPF+WF+WCS .Net 3.5=.Net 3 ...

  2. 前端向后台的华丽转身 — PHP入门篇

    三个月就这么悄悄溜走了,本K对于前端虽然有了一定的认识,但对一些方面还是处于一种比较萌币的状态,就在这种萌币状态下,本K又跟着大神浩开始了后台语言-PHP语言的学习.PHP的学习对于学过其他语言的人来 ...

  3. 【webpack】webpack-dev-server生猛上手——让我们来搭一个webpack的微服务器吧!

      [前言]:因为最近在搞百度地图API的时候到了webpack的externals,才发现我之前都只是用webpack做一些搭建完项目后的"收尾工作"--即打包,而没有把它纳入到 ...

  4. (转) Java RMI 框架(远程方法调用)

    "原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 .作者信息和本声明.否则将追究法律责任.http://haolloyin.blog.51cto.com/1177454/33 ...

  5. Java常用类之要点总结

    Java常用类之要点总结

  6. linux--centos服务器配置

    配置Centos服务器   1. 安装centos 安装时注意分区留下一个/data区给网站内容存储. 2. 配置网络连接 默认centos网络连接不是开机启动的,需要配置. vi /etc/sysc ...

  7. ap.net core 教程(三) - 新建项目

    ASP.NET Core - 新建项目 在这一章,我们将讨论如何在Visual Studio中创建一个新项目. 只要你安装了Visual Studio 2015的.net core工具,您就可以开始构 ...

  8. kali高速更新源以及主题修改方法

    文章不小心删了~这是我以前写的文章了了.实用性较强,所以现在补回来! 安装完kali之后,需要对软件进行一次整体更新:apt-get update & apt-get upgrade 但是,先 ...

  9. xfire调用webservice接口的实现方式

    package com.test; import java.net.URL; import org.codehaus.xfire.client.Client; import org.codehaus. ...

  10. Deep Q-Network 学习笔记(一)—— Q-Learning 学习与实现过程中碰到的一些坑

    这方面的资料比较零散,学起来各种碰壁,碰到各种问题,这里就做下学习记录. 参考资料: https://morvanzhou.github.io/ 非常感谢莫烦老师的教程 http://mnemstud ...