poj1741 树上距离小于等于k的对数 点分治 入门题
#include <iostream>
#include <stdio.h>
#include <string.h>
#include <algorithm>
#define N 40005
#define M 80005
#define LL long long
using namespace std;
const int INF = 0x3f3f3f3f;
int ver[M],edge[M],head[N],Next[M];
int n,m,tot,root;LL k;
void add(int x,int y,int w){
ver[++tot]=y;edge[tot]=w;Next[tot]=head[x];head[x]=tot;
ver[++tot]=x;edge[tot]=w;Next[tot]=head[y];head[y]=tot;
}
int sz[N],vis[N],mx,size;
LL d[N],q[N],l,r,ans=0;
//求出树的重心 因为找到重心后,递归子树不超过原来的一半,递归层数小于logn层
void getroot(int u,int fa){
sz[u]=1;int num=0;
for (int i=head[u];i;i=Next[i]){
int v=ver[i];
if (v==fa||vis[v])continue;
///继续深搜
getroot(v,u);
///计算出子树的大小
sz[u]+=sz[v];
///维护子树的最长的链
num=max(num,sz[v]);
}
///num代表的是子节点的最长链 size-sz[u]代表的是父亲链长
num=max(num,size-sz[u]);
if (num<mx)mx=num,root=u;
}
///计算某个节点所有子树中的节点的到这个节点的距离
void getdis(int u,int fa){
q[++r]=d[u];
for(int i=head[u];i;i=Next[i]){
int v=ver[i];
if(v==fa||vis[v])continue;
d[v]=d[u]+edge[i];
getdis(v,u);
}
}
LL cal(int u,int val){
r=0;
d[u]=val;
getdis(u,0);
LL sum=0,l=1;
///把子树点到当前点的距离进行排序
sort(q+1,q+1+r);
cout<<u<<" "<<r<<endl;
for(int i=1;i<=r;i++){
cout<<q[i]<<" ";
}
cout<<endl;
///开一个左右指针,以左端点为基准移动,如果两个距离是大于k,肯定移动右指针,
///也就是对于每一个小的,去右边寻找最远能满足条件的,而中间的一定满足
while(l<r){
if(q[l]+q[r]<=k)sum+=r-l,++l;
else --r;
}
return sum;
}
void dfs(int u){
///计算当前节点内部所有>=k的数目 但是会存在连个点是在同一联通块内部 答案就不对了
ans+=cal(u,0);
vis[u]=1;
for (int i=head[u];i;i=Next[i]){
int v=ver[i];
if (vis[v])continue;
///这里我们减去内部在同一个联通块里面的答案 相当于剪掉重复的
ans-=cal(v,edge[i]);
///在当前点内部继续找重心
size=sz[v];
mx=INF;
getroot(v,0);
///然后找到子树的重心,进行深搜
dfs(root);
}
}
int main(){
int u,v,e,k;
while(~scanf("%d%d",&n,&k) && n+k) {
ans=0;
memset(vis,0,sizeof(vis));
memset(head,0,sizeof(head));
memset(Next,0, sizeof(Next));
for (int i = 1; i <n ; i++) {
scanf("%d%d%d", &u, &v, &e);
add(u, v, e);
}
size = n;
mx = INF;
getroot(1, 0);
dfs(root);
printf("%lld\n", ans);
}
return 0;
}
poj1741 树上距离小于等于k的对数 点分治 入门题的更多相关文章
- POJ1741--Tree (树的点分治) 求树上距离小于等于k的点对数
Tree Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 12276 Accepted: 3886 Description ...
- P3806 离线多次询问 树上距离为K的点对是否存在 点分治
询问树上距离为k的点对是否存在 直接n^2暴力处理点对 桶排记录 可以过 #include<cstdio> #include<cstring> #include<algo ...
- Codeforces 161.D. Distance in Tree-树分治(点分治,不容斥版)-树上距离为K的点对数量-蜜汁TLE (VK Cup 2012 Round 1)
D. Distance in Tree time limit per test 3 seconds memory limit per test 512 megabytes input standard ...
- 洛谷 P3806 【模板】点分治1-树分治(点分治,容斥版) 模板题-树上距离为k的点对是否存在
P3806 [模板]点分治1 题目背景 感谢hzwer的点分治互测. 题目描述 给定一棵有n个点的树 询问树上距离为k的点对是否存在. 输入格式 n,m 接下来n-1条边a,b,c描述a到b有一条长度 ...
- [HNOI2003]消防局的设立(树上距离为k的最小覆盖问题)
题目的大概意思现在有一棵树,在树上找半径小于等于2的最小覆盖点的最小个数. 题目链接 讲一讲此类题的贪心策略: 就是每次寻找最低没有被覆盖的点,显然对于覆盖它的所有点中,在他的祖先处设立一个点最优.所 ...
- poj 1987 节点距离小于等于K(树DP)
这题和poj 1741是一模一样的 但是1741能AC的代码,在这里却是TLE,暂时没看出哪里出现了问题.. AC代码: #include <iostream> #include < ...
- 树点分治入门题poj1741
Tree Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 24253 Accepted: 8060 Description ...
- poj1741 树上的分治
题意是说给了n个点的树n<=10000,问有多少个点对例如(a,b)他们的之间的距离小于等于k 采用树的分治做 #include <iostream> #include <cs ...
- 【点分治】【路径小于等于k的条数】【路径恰好等于k是否存在】
POJ1741:Tree Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 29574 Accepted: 9915 Des ...
随机推荐
- 删除 java代码中所有的注释
删除 java代码中所有的注释.java public class CleanCommons { private static Pattern pattern = Pattern.compile(&q ...
- python实例 输出字符串和数字
但有趣的是,在javascript里我们会理想当然的将字符串和数字连接,因为是动态语言嘛.但在Python里有点诡异,如下: #! /usr/bin/python a=2 b="test&q ...
- H5C3--sessionStorage和localStorage的使用
一.sessionStorage的使用 <!DOCTYPE html> <html lang="en"> <head> <meta cha ...
- HDU 1536 求解SG函数
#include<stdio.h> #include<string.h> #include<algorithm> #include<set> using ...
- twisted(转)
reactor.protocol 这两个类都在 twisted.internet 命名空间中 reactor对象是Twisted编程当中的第一步,它就是一个反应器,专门负责与服务端的连接以及监听与服务 ...
- Ajax--Ajax基于原生javascript:创建Ajax对象、链接服务器、发送请求、接受响应结果
Ajax概述 异步:指某段程序执行时不会阻塞其它程序执行,其表现形式为程序的执行顺序不依赖程序本身的书写顺序,相反则为同步. 同步请求: 请求是由浏览器发送 页面会刷新 异步请求: 请求是由浏览器的一 ...
- SPARK Day04
广播变量和累加器 广播变量 广播变量理解图 广播变量使用 val conf = new SparkConf() conf.setMaster("local").setAppName ...
- ACM:树的变换,依据表达式建立表达式树
题目:输入一个表达式.建立一个表达式树. 分析:找到最后计算的运算符(它是整棵表达式树的根),然后递归处理! 在代码中.仅仅有当p==0的时候.才考虑这个运算符,由于括号中的运 ...
- apply( )与 call( ) 的区别
JavaScript中的每一个Function对象都有一个apply()方法和一个call()方法 语法 /*apply()方法*/ function.apply(thisObj[, argArray ...
- jq获取浏览器可视窗口的高度
<script> var window_height = $(window).height(); </script>