点分治


怎么又一道叫Tree的题目……真是醉了。

本题为漆子超论文《分治算法在树的路径问题中的应用》例一

题解 : http://blog.csdn.net/sdj222555/article/details/7893862

       http://blog.csdn.net/yang_7_46/article/details/9966455

  既然是点分治嘛,为了保证复杂度不退化,必然要找树的重心,这一步可以通过一次dfs实现:以任意节点为根(方便起见就选1号节点了)求出每个节点子树大小,那么当前节点就将整棵树分为size[x],n-size[x]两部分,我们取较大的那一部分作为当前节点的子树大小(只是记录个值而已, 不必在意具体哪个是根),然后我们可以在dfs到每个节点时与全局变量root比较一下 f[x]和f[root] 如果f[x]<f[root] 则root=x(这里用root表示重心)(由于f[x]>=n/2,所以f[x]最小的那个x就是将整棵树分的最平均的那个“重心”)

  找到重心后,需要统计所有点(n-1个)到重心的距离,统计有多少对点之间的距离小于等于k。同样用一次dfs实现,然后将所有节点到重心的距离存在一个数组里,进行一次快排~然后利用单调性,一次相向搜索在O(n)时间内解决。但是这是有额外计数的:两点在同一子树里的是不能算的,所以减去两点在同一子树里的情况(递归求解子树的时候会再算一次)。当然还是要递归算各子树内的点对啦~

 //POJ 1741
#include<cstdio>
#include<vector>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<algorithm>
#define rep(i,n) for(int i=0;i<n;++i)
#define F(i,j,n) for(int i=j;i<=n;++i)
#define D(i,j,n) for(int i=j;i>=n;--i)
#define pb push_back
using namespace std;
void read(int &v){
v=; int sign=; char ch=getchar();
while(ch<''||ch>''){ if (ch=='-') sign=-; ch=getchar();}
while(ch>=''&&ch<=''){ v=v*+ch-''; ch=getchar();}
v*=sign;
}
/******************tamplate*********************/
const int N=;
struct edge{
int to,l;
};
int n,k,size,s[N],f[N],root,d[N],ans,K;
vector<edge>G[N];
vector<int>dep;
bool vis[N]; void getroot(int x,int fa){
int y;
s[x]=;f[x]=;
rep(i,G[x].size()){
y=G[x][i].to;
if (y!=fa && !vis[y]){
getroot(y,x);
s[x]+=s[y];
f[x]=max(f[x],s[y]);
}
}
f[x]=max(f[x],size-s[x]);
if (f[x]<f[root]) root=x;
}
void getdep(int x,int fa){
int y;
dep.pb(d[x]);
s[x]=;
rep(i,G[x].size()){
y=G[x][i].to;
if (y!=fa && !vis[y]){
d[y]=d[x]+G[x][i].l;
getdep(y,x);
s[x]+=s[y];
}
}
}
int calc(int x,int init){
dep.clear(); d[x]=init;
getdep(x,);
sort(dep.begin(),dep.end());
int ans=;
for(int l=,r=dep.size()-; l<r; )
if (dep[l]+dep[r]<=k) ans+=r-l++;
else r--;
return ans;
}
void work(int x){
int y;
ans+=calc(x,);
vis[x]=;
rep(i,G[x].size()){
y=G[x][i].to;
if (!vis[y]){
ans-=calc(y,G[x][i].l);
f[]=size=s[y];
getroot(y,root=);
work(root);
}
}
}
int main(){
// freopen("1741.in","r",stdin);
while(scanf("%d%d",&n,&k)==){
if (n== && K==) break;
int x,y,z;
F(i,,n) G[i].clear();
memset(vis,,sizeof vis);
F(i,,n){
read(x); read(y); read(z);
G[x].pb((edge){y,z}); G[y].pb((edge){x,z});
}
f[]=size=n;
getroot(,root=);
ans=;
work(root);
printf("%d\n",ans);
}
return ;
}

1468: Tree

Time Limit: 10 Sec  Memory Limit: 64 MB
Submit: 606  Solved: 317
[Submit][Status][Discuss]

Description

给你一棵TREE,以及这棵树上边的距离.问有多少对点它们两者间的距离小于等于K

Input

N(n<=40000)
接下来n-1行边描述管道,按照题目中写的输入
接下来是k

Output

一行,有多少对点之间的距离小于等于k

Sample Input

7
1 6 13
6 3 9
3 5 7
4 1 3
2 4 20
4 7 2
10

Sample Output

5

HINT

Source

[Submit][Status][Discuss]

【POJ】【1741】/【BZOJ】【1468】Tree的更多相关文章

  1. bzoj 2295: 【POJ Challenge】我爱你啊

    2295: [POJ Challenge]我爱你啊 Time Limit: 1 Sec  Memory Limit: 128 MB Description ftiasch是个十分受女生欢迎的同学,所以 ...

  2. 【链表】BZOJ 2288: 【POJ Challenge】生日礼物

    2288: [POJ Challenge]生日礼物 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 382  Solved: 111[Submit][S ...

  3. BZOJ 2287: 【POJ Challenge】消失之物( 背包dp )

    虽然A掉了但是时间感人啊.... f( x, k ) 表示使用前 x 种填满容量为 k 的背包的方案数, g( x , k ) 表示使用后 x 种填满容量为 k 的背包的方案数. 丢了第 i 个, 要 ...

  4. 【POJ 1741】Tree

    Tree Time Limit: 1000MS   Memory Limit: 30000K Total Submissions: 11570   Accepted: 3626 Description ...

  5. bzoj 2288 【POJ Challenge】生日礼物 双向链表+堆优化

    2288: [POJ Challenge]生日礼物 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1003  Solved: 317[Submit][ ...

  6. 【BZOJ 2288】 2288: 【POJ Challenge】生日礼物 (贪心+优先队列+双向链表)

    2288: [POJ Challenge]生日礼物 Description ftiasch 18岁生日的时候,lqp18_31给她看了一个神奇的序列 A1, A2, ..., AN. 她被允许选择不超 ...

  7. BZOJ 2287 【POJ Challenge】消失之物(DP+容斥)

    2287: [POJ Challenge]消失之物 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 986  Solved: 572[Submit][S ...

  8. 【poj 1984】&【bzoj 3362】Navigation Nightmare(图论--带权并查集)

    题意:平面上给出N个点,知道M个关于点X在点Y的正东/西/南/北方向的距离.问在刚给出一定关系之后其中2点的曼哈顿距离((x1,y1)与(x2,y2):l x1-x2 l+l y1-y2 l),未知则 ...

  9. bzoj2287【POJ Challenge】消失之物 缺一01背包

    bzoj2287[POJ Challenge]消失之物 缺一01背包 链接 bzoj 思路 分治solve(l,r,arr)表示缺少物品\([l,r]\)的dp数组arr. 然后solve(l,mid ...

  10. 【poj 1988】Cube Stacking(图论--带权并查集)

    题意:有N个方块,M个操作{"C x":查询方块x上的方块数:"M x y":移动方块x所在的整个方块堆到方块y所在的整个方块堆之上}.输出相应的答案. 解法: ...

随机推荐

  1. js利用数组length属性清空和截短数组

    1.使用length清空数组: 代码如下 复制代码 <script>    var arr1 = ['aaa','bbbb','http://www.111cn.net'];    ale ...

  2. QtPropertyBrowser+vs2010的安装与配置(转)

    这一篇文章有些问题,后又写了一篇,地址是http://www.cnblogs.com/aminxu/p/4552410.html 转自http://blog.csdn.net/jingwenlai_s ...

  3. Objective-C设计模式——工厂方法模式virtual constructor(对象创建)

    工厂方法模式 工厂方法模式可以控制对象的创建过程,屏蔽对象创建的细节,可以直接创建出我们所需要的已经配置好的对象. 工厂方法模式定义了创建方法的接口,让子类决定实例化哪一个类,工厂方法模式使得一个类的 ...

  4. Revit二次开发-BIM模型导出

    最近一个月一直在研究Revit二次开发-BIM模型的导出,在网上找了很多相关资料学习.下面简单介绍一下我最近做的这个BIM模型的导出功能. 开始尝试使用Revit2015的样例程序里提供的读取模型几何 ...

  5. 【学习】Windows PE文件学习(一:导出表)

    今天做了一个读取PE文件导出表的小程序,用来学习. 参考了<Windows PE权威指南>一书. 首先, PE文件的全称是Portable Executable,可移植的可执行的文件,常见 ...

  6. 大数求模 sicily 1020

        Search

  7. IOS基础之设置APP的名字、设置图标、添加等待加载时的图片

    1.我们的app一般默认的名字是我们的工程名字,所以我们一般会更改一个更加友好的名字,更改的方法如下图: 找到InfoPlist.strings文件,在里面添加语句CFBundleDisplayNam ...

  8. 【Sharing】开发与研发

    [声明]此文为转载,只为收藏. 按:这几天我一直在写这篇东西,本来是胸有成竹,没想到后来越写越发现自己在这个题目下有太多话想说,而以我现在的能力又不能很好地概括总结,以至于越写越长,文章结构也变得混乱 ...

  9. yum被锁定

    使用Yum的时候 提示yum被搜定了 . Another app is currently holding the yum lock; waiting for it to exit...   解决办法 ...

  10. $.each遍历json对象

    查看一个简单的jQuery的例子来遍历一个JavaScript数组对象. var json = [ {"id":"1","tagName": ...