Description

Give a tree with n vertices,each edge has a length(positive integer less than 1001). 
Define dist(u,v)=The min distance between node u and v. 
Give an integer k,for every pair (u,v) of vertices is called valid if and only if dist(u,v) not exceed k. 
Write a program that will count how many pairs which are valid for a given tree. 
            --by  POJ
http://poj.org/problem?id=1741


点分治的相关题目;
有关点分治的内容;
题目大意为查询树上最短距离不大于k的点对个数;
我们应用点分治的思路,
统计在完全在某子树内经过其重心的路径;
然后再递归分治下去;
统计方式采取,dfs出子树中,点到重心的距离,查找合法(dis(i)+dis(j)≤k)的组合;
如何查找合法的组合?
 sort(dis+,dis+num+);
r=num;
for(l=;l<=num,l<r;l++){
while(dis[l]+dis[r]>k&&l<r)
  r--;
ans+=r-l;
}

可以看出查找效率主要在排序上,反正很快;

但发现当两点在同一子树中且dis(i)+dis(j)≤k时,他们被计入答案,但他们的最短路径甚至不经过该重心;

于是把这些点用相似的方式查出再减去即可;

代码:

#include<cstdio>
#include<algorithm>
#include<cstring>
using namespace std;
struct ss{
int next,to,w;
};
ss e[];
int first[],num;
int n,k,ans,size[],max_size[],vis[],dis[];
inline void Init();
inline void build(int ,int ,int );
inline void part_dfs(int );
inline void dfs_size(int ,int );
inline int dfs_root(int ,int ,int );
inline int chan_ans(int ,int );
inline void dfs_len(int ,int ,int );
inline void in(int &ans)
{
ans=;bool p=false;char ch=getchar();
while((ch>'' || ch<'')&&ch!='-') ch=getchar(); if(ch=='-') p=true,ch=getchar();
while(ch<=''&&ch>='') ans=ans*+ch-'',ch=getchar(); if(p) ans=-ans;
}
int main()
{
int i,j,u,v,w;
while(scanf("%d%d",&n,&k)==){
if(n==)
return ;
Init();
for(i=;i<=n-;i++){
in(u),in(v),in(w);
build(u,v,w);build(v,u,w);
}
part_dfs();
printf("%d\n",ans);
}
}
inline void Init(){
memset(vis,,sizeof(vis));
memset(first,,sizeof(first));
ans=;num=;
}
inline void build(int f,int t,int wi){
e[++num].next=first[f];first[f]=num;
e[num].to=t;e[num].w=wi;
}
inline void part_dfs(int now){
int root,i;
dfs_size(now,);
root=dfs_root(now,,now);
ans+=chan_ans(root,);
vis[root]=;
for(i=first[root];i;i=e[i].next)
if(!vis[e[i].to]){
ans-=chan_ans(e[i].to,e[i].w);
part_dfs(e[i].to);
}
}
inline void dfs_size(int now,int fa){
size[now]=;
for(int i=first[now];i;i=e[i].next)
if(!vis[e[i].to]&&e[i].to!=fa){
dfs_size(e[i].to,now);
size[now]+=size[e[i].to];
}
}
inline int dfs_root(int now,int fa,int r){
int i,root=-,wroot;
max_size[now]=size[r]-size[now];
for(i=first[now];i;i=e[i].next)
if(!vis[e[i].to]&&e[i].to!=fa){
if(size[e[i].to]>max_size[now])
max_size[now]=size[e[i].to];
wroot=dfs_root(e[i].to,now,r);
if(max_size[wroot]<max_size[root]||root==-)
root=wroot;
}
if(max_size[now]<max_size[root]||root==-)
root=now;
return root;
}
inline int chan_ans(int root,int dis1){
int l,r,ans=;
num=;
dfs_len(root,dis1,);
sort(dis+,dis+num+);
r=num;
for(l=;l<=num,l<r;l++){
while(dis[l]+dis[r]>k&&l<r)r--;
ans+=r-l;
}
return ans;
}
inline void dfs_len(int now,int d,int fa){
dis[++num]=d;
for(int i=first[now];i;i=e[i].next)
if(!vis[e[i].to]&&e[i].to!=fa)
dfs_len(e[i].to,d+e[i].w,now);
}
//点分治:
//dfs分治{
//对每个分支dfs(两遍)重心
//以重心为根dfs统计合法路径个数
//}分治重心的子树结构
//dis:一个不具有顺序的点到当前结构的重心的距离表
//vis[i]标记(染色)已到过的重心
//size[i]
//max_size[i]记录某点的最大子树大小;
//由于基于每个分支的dfs之间是跳着的(因为现在的重心未必是前重心的子节点),所以需要vis顺便截断某次dfs,防止跑出分支

祝AC

POJ P1741 Tree 解题报告的更多相关文章

  1. 【LeetCode】863. All Nodes Distance K in Binary Tree 解题报告(Python)

    [LeetCode]863. All Nodes Distance K in Binary Tree 解题报告(Python) 作者: 负雪明烛 id: fuxuemingzhu 个人博客: http ...

  2. 【LeetCode】297. Serialize and Deserialize Binary Tree 解题报告(Python)

    [LeetCode]297. Serialize and Deserialize Binary Tree 解题报告(Python) 标签: LeetCode 题目地址:https://leetcode ...

  3. 【LeetCode】331. Verify Preorder Serialization of a Binary Tree 解题报告(Python)

    [LeetCode]331. Verify Preorder Serialization of a Binary Tree 解题报告(Python) 标签: LeetCode 题目地址:https:/ ...

  4. 【LeetCode】109. Convert Sorted List to Binary Search Tree 解题报告(Python)

    [LeetCode]109. Convert Sorted List to Binary Search Tree 解题报告(Python) 标签(空格分隔): LeetCode 作者: 负雪明烛 id ...

  5. 【LeetCode】236. Lowest Common Ancestor of a Binary Tree 解题报告(Python)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 日期 题目地址:https://leetcode.c ...

  6. 【LeetCode】99. Recover Binary Search Tree 解题报告(Python)

    [LeetCode]99. Recover Binary Search Tree 解题报告(Python) 标签(空格分隔): LeetCode 题目地址:https://leetcode.com/p ...

  7. 【LeetCode】662. Maximum Width of Binary Tree 解题报告(Python)

    [LeetCode]662. Maximum Width of Binary Tree 解题报告(Python) 标签(空格分隔): LeetCode 题目地址:https://leetcode.co ...

  8. 【LeetCode】623. Add One Row to Tree 解题报告(Python)

    [LeetCode]623. Add One Row to Tree 解题报告(Python) 标签(空格分隔): LeetCode 题目地址:https://leetcode.com/problem ...

  9. POJ 2054 Color a Tree解题报告

    题干 Bob is very interested in the data structure of a tree. A tree is a directed graph in which a spe ...

随机推荐

  1. clang命令理解程序

    Xcode 创建一个mac OS   command Line Tool程序 步骤打开终端  cd + 工程路径(绝对路径)(注:拖拽main.m文件到终端) input —preprocessor— ...

  2. iOS学习笔记(8)——GCD初探

    1. AppDelegate.m #import "AppDelegate.h" #import "ViewController.h" @interface A ...

  3. iOS学习笔记(5)——显示简单的TableView

    1. 创建工程 创建一个新的Xcode工程命名为SimpleTableTest. 删除main.storyboard文件和info.plist中有关storyboard的相关属性. 按command+ ...

  4. Echarts【1、数据过多导致显示不全分页,2、数据展示探讨分析】

    var len=<c:out value="${len }"></c:out>; var dataZoom_end=null; //为空默认100%所以默认 ...

  5. Python 全栈开发:dict(字典)常用方法操作、dict嵌套

    数据类型的划分:可变数据类型和不可变数据类型. 不可变数据类型(可哈希):元祖.bool.int.str 可变数据类型(不可哈希):list.dict,set(集合) dict(字典): dict(字 ...

  6. 【GIS新探索】算法实现在不规则区域内均匀分布点

    1 概要 在不规则区域内均匀分布点,这个需求初看可能不好理解.如果设想一下需求场景就比较简单了. 场景1:在某个地区范围内,例如A市区有100W人口,需要将这100W人口在地图上面相对均匀的标识出来. ...

  7. java容器类1:Collection,List,ArrayList,LinkedList深入解读

    1. Iterable 与 Iterator Iterable 是个接口,实现此接口使集合对象可以通过迭代器遍历自身元素. public interface Iterable<T> 修饰符 ...

  8. Chrome DevTools的15个使用技巧(译)

    谷歌浏览器如今是Web开发者们所使用的最流行的网页浏览器.伴随每六个星期一次的发布周期和不断扩大的强大的开发功能,Chrome变成了一个必须掌握的工具.大多数前端开发者可能熟悉关于chorme的许多特 ...

  9. javac之Method Invocation Expressions

    15.12.1. Compile-Time Step 1: Determine Class or Interface to Search 15.12.2. Compile-Time Step 2: D ...

  10. 关于Input输入框蓝色外框的操作

    1.去掉input外框的css: input { outline : none; //去掉外框 //outline:medium; } input:focus { outline : none; } ...