浅谈树分治:https://www.cnblogs.com/AKMer/p/10014803.html

题目传送门:http://poj.org/problem?id=1741

这是一道树分治的模板题。

我们考虑当前经过联通块的重心\(rt\)的路径小于等于\(k\)的有多少条,不经过\(rt\)的等把\(rt\)这个点删了之后再递归去统计。

我们可以记录一下从\(rt\)出发到每个点的距离\(dis\),然后把\(dis\)从小到大排序,用两个指针\(l\)和\(r\)分别指向\(dis\)数组的开头和结尾。每次以\(l\)为基准,找能与\(l\)匹配起来小于等于\(k\)的另一条路径。假设\(l\)的\(dis\)加上\(r\)的\(dis\)小于等于\(k\),且\(r\)是满足这个条件的最长的路径,那么区间\([l,r]\)里的所有\(dis\)加上\(l\)的\(dis\)都会小于等于\(k\)。所以对于\(l\),能与它匹配的路径条数是\(r-l+1-num\),\(num\)表示区间\([l,r]\)中与\(l\)在同一个子树里的路径条数,这个我们可以开个桶动态维护。然后对于\(l\)不断增加,\(r\)必然是递减的,所以复杂度是\(O(n)\)的。因为每个点只会被递归\(log\)次,所以总复杂度是\(O(nlog^2n)\)的。

边分治做法差不太多,不过空间会因为要重建树而翻一倍。

时间复杂度:\(O(nlog^2n)\)

空间复杂度:\(O(n)\)

点分治版代码如下:

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std; const int maxn=1e4+5; bool vis[maxn];
int siz[maxn],num[maxn];
int n,limit,tot,ans,mx,rt,cnt,N;
int now[maxn],son[maxn*2],pre[maxn*2],val[maxn*2]; int read() {
int x=0,f=1;char ch=getchar();
for(;ch<'0'||ch>'9';ch=getchar())if(ch=='-')f=-1;
for(;ch>='0'&&ch<='9';ch=getchar())x=x*10+ch-'0';
return x*f;
} struct road {
int bel,dis; road() {} road(int _bel,int _dis) {
bel=_bel,dis=_dis;
} bool operator<(const road &a)const {
return dis<a.dis;
}
}tmp[maxn]; void add(int a,int b,int c) {
pre[++tot]=now[a];
now[a]=tot,son[tot]=b,val[tot]=c;
} void clear() {
ans=tot=0;
memset(now,0,sizeof(now));
memset(vis,0,sizeof(vis));
} void find_root(int fa,int u) {
int res=0;siz[u]=1;
for(int p=now[u],v=son[p];p;p=pre[p],v=son[p])
if(!vis[v]&&v!=fa)
find_root(u,v),siz[u]+=siz[v],res=max(res,siz[v]);
res=max(res,N-siz[u]);
if(res<mx)mx=res,rt=u;
} void solve(int belong,int fa,int u,int dis) {
tmp[++cnt]=road(belong,dis);siz[u]=1;//之前求出的siz是以u为根的,现在顺便把以rt为根时的siz求出来,递归下去就直接可以调用了。
for(int p=now[u],v=son[p];p;p=pre[p],v=son[p])
if(!vis[v]&&v!=fa)
solve(belong,u,v,dis+val[p]),siz[u]+=siz[v];
} void point_division(int u,int size) {//size表示当前联通块总大小
N=size,mx=rt=n+1,find_root(0,u);
u=rt,cnt=0,num[0]=1,vis[u]=1;
for(int p=now[u],v=son[p];p;p=pre[p],v=son[p])
if(!vis[v])solve(v,u,v,val[p]),num[v]=siz[v];
sort(tmp+1,tmp+cnt+1);int l=0,r=cnt;
while(l<r) {
while(r>l&&tmp[l].dis+tmp[r].dis>limit)num[tmp[r].bel]--,r--;
if(l<r)ans+=(r-l+1)-num[tmp[l].bel];num[tmp[l].bel]--,l++;
}
for(int p=now[u],v=son[p];p;p=pre[p],v=son[p])
if(!vis[v])point_division(v,siz[v]);
} int main() {
while(1) {
n=read(),limit=read();
if(!n)break; clear();
for(int i=1;i<n;i++) {
int x=read(),y=read(),v=read();
add(x,y,v),add(y,x,v);
}
point_division(1,n);
printf("%d\n",ans);
}
return 0;
}

边分治版代码如下:

#include <cmath>
#include <vector>
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
typedef pair<int,int> pii;
#define fr first
#define sc second const int maxn=2e4+5; bool vis[maxn];
int siz[maxn],num[maxn];
int n,limit,tot=1,ans,mx,id,cnt,N;
int now[maxn],pre[maxn*2],son[maxn*2],val[maxn*2]; vector<pii>to[maxn];
vector<pii>::iterator it; int read() {
int x=0,f=1;char ch=getchar();
for(;ch<'0'||ch>'9';ch=getchar())if(ch=='-')f=-1;
for(;ch>='0'&&ch<='9';ch=getchar())x=x*10+ch-'0';
return x*f;
} struct road {
int bel,dis; road() {} road(int _bel,int _dis) {
bel=_bel,dis=_dis;
} bool operator<(const road &a)const {
return dis<a.dis;
}
}tmp[maxn]; void clear() {
tot=ans=0;
memset(vis,0,sizeof(vis));
memset(now,0,sizeof(now));
} void add(int a,int b,int c) {
pre[++tot]=now[a];
now[a]=tot,son[tot]=b,val[tot]=c;
} void find_son(int fa,int u) {
to[u].clear();
for(int p=now[u],v=son[p];p;p=pre[p],v=son[p])
if(v!=fa)to[u].push_back(make_pair(v,val[p])),find_son(u,v);
} void rebuild() {
tot=1,memset(now,0,sizeof(now));
for(int i=1;i<=cnt;i++) {
int size=to[i].size();
if(size<=2) {
for(it=to[i].begin();it!=to[i].end();it++) {
pii tmp=*it;
add(i,tmp.fr,tmp.sc),add(tmp.fr,i,tmp.sc);
}
}
else {
pii u1=make_pair(++cnt,0),u2;
if(size==3)u2=to[i].front();else u2=make_pair(++cnt,0);
add(i,u1.fr,u1.sc),add(u1.fr,i,u1.sc);
add(i,u2.fr,u2.sc),add(u2.fr,i,u2.sc);
if(size>3) {
to[cnt-1].clear();to[cnt].clear();int tmp=0;
for(it=to[i].begin();it!=to[i].end();it++) {
if(!tmp)to[cnt].push_back(*it);
else to[cnt-1].push_back(*it);tmp^=1;
}
}
else {
to[cnt].clear();
for(int j=1;j<=2;j++)
to[cnt].push_back(to[i].back()),to[i].pop_back();
}
}
}
} void find_edge(int fa,int u) {
siz[u]=1;
for(int p=now[u],v=son[p];p;p=pre[p],v=son[p])
if(!vis[p>>1]&&v!=fa) {
find_edge(u,v),siz[u]+=siz[v];
if(abs(N-2*siz[v])<mx)mx=abs(N-2*siz[v]),id=p>>1;
}
} void solve(int bel,int fa,int u,int dis) {
if(u<=n)tmp[++tot]=road(bel,dis),num[bel]++;siz[u]=1;
for(int p=now[u],v=son[p];p;p=pre[p],v=son[p])
if(!vis[p>>1]&&v!=fa)solve(bel,u,v,dis+val[p]),siz[u]+=siz[v];
} void work(int u,int size) {
N=size,mx=id=cnt+1,find_edge(0,u);
tot=0,vis[id]=1;if(id==cnt+1)return;
int u1=son[id<<1],u2=son[id<<1|1];
num[u1]=num[u2]=0;
solve(u1,0,u1,0),solve(u2,0,u2,0);
sort(tmp+1,tmp+tot+1);int l=1,r=tot;
while(l<r) {
while(r>l&&tmp[r].dis+tmp[l].dis>limit-val[id<<1])num[tmp[r].bel]--,r--;
if(l<r)ans+=(r-l+1)-num[tmp[l].bel];num[tmp[l].bel]--,l++;
}
work(u1,siz[u1]),work(u2,siz[u2]);
} int main() {
while(1) {
cnt=n=read();limit=read();
if(!n)break; clear();
for(int i=1;i<n;i++) {
int x=read(),y=read(),v=read();
add(x,y,v),add(y,x,v);
}
find_son(0,1);rebuild();
work(1,cnt);printf("%d\n",ans);
}
return 0;
}

POJ1741:Tree的更多相关文章

  1. POJ1741:Tree——题解+树分治简要讲解

    http://poj.org/problem?id=1741 题目大意:给一棵树,求点对间距离<=k的个数. ———————————————————— 以这道题为例记录一下对于树分治的理解. 树 ...

  2. 【POJ1741】Tree(点分治)

    [POJ1741]Tree(点分治) 题面 Vjudge 题目大意: 求树中距离小于\(K\)的点对的数量 题解 完全不觉得点分治了.. 简直\(GG\),更别说动态点分治了... 于是来复习一下. ...

  3. codeforces 375D:Tree and Queries

    Description You have a rooted tree consisting of n vertices. Each vertex of the tree has some color. ...

  4. POJ 3237:Tree(树链剖分)

    http://poj.org/problem?id=3237 题意:树链剖分.操作有三种:改变一条边的边权,将 a 到 b 的每条边的边权都翻转(即 w[i] = -w[i]),询问 a 到 b 的最 ...

  5. leetcode题解:Tree Level Order Traversal II (二叉树的层序遍历 2)

    题目: Given a binary tree, return the bottom-up level order traversal of its nodes' values. (ie, from ...

  6. 命令行打印文件树列表: tree

    Linux & Mac 1.下载tree lib //mac brew install tree //centos yum install tree //ubuntu apt-get inst ...

  7. Linux:tree命令详解

    tree 以树状图列出目录的内容 语法 tree(选项)(参数) 选项 -a:显示所有文件和目录: -A:使用ASNI绘图字符显示树状图而非以ASCII字符组合: -C:在文件和目录清单加上色彩,便于 ...

  8. TYOI Day1 travel:Tree dp【处理重复走边】

    题意: 给你一棵树,n个节点,每条边有长度. 然后有q组询问(u,k),每次问你:从节点u出发,走到某个节点的距离mod k的最大值. 题解: 对于无根树上的dp,一般都是先转成以1为根的有根树,然后 ...

  9. linux命令学习笔记(61):tree 命令

    shendu@shenlan:~$ tree 程序“tree”尚未安装. 您可以使用以下命令安装: sudo apt-get install tree shendu@shenlan:~$ sudo a ...

随机推荐

  1. 转: WebRTC音视频引擎研究(1)--整体架构分析

    转自: http://blog.csdn.net/temotemo/article/details/7530504   目录(?)[+]   WebRTC技术交流群:234795279 原文地址:ht ...

  2. 计算机的一些经典书籍CS经典书单

    c++: <c++程序设计> <c++primer> <effective c++> <more effective c++> <深入探索c++对 ...

  3. python正则中的贪婪与非贪婪

    当重复一个正则表达式时,如用 a*,操作结果是尽可能多地匹配模式.当你试着匹配一对对称的定界符,如 HTML 标志中的尖括号.匹配单个 HTML 标志的模式不能正常工作,因为 .* 的本质是“贪婪”的 ...

  4. 实习日记)select option 选择不同的option时, 页面发生不同的变化

    怎么在下拉框的选择不同的option时, 页面发生响应的变化 因为option是没有点击事件什么的,  只有select才有, 所以不能通过option的点击事件来完成, 所以开始的尝试都失败了(之前 ...

  5. erlang启动参数记录

    不管在erlang的shell下还是脚本里,启动参数都是非常有用的,抽空儿整理下erlang的常用启动参数: +A size   异步线程池的线程数,范围为0~1024,默认为10 +P Number ...

  6. Clustering of residential areas based on residential conditions

    watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQveWFuZ3hpYW5neXVpYm0=/font/5a6L5L2T/fontsize/400/fill/I0 ...

  7. 开源流媒体服务器EasyDarwin支持epoll网络模型,大大提升流媒体服务器网络并发性能

    经过春节前后将近2个月的开发和稳定调试.测试,EasyDarwin开源流媒体服务器终于成功将底层select网络模型修改优化成epoll网络模型,将EasyDarwin流媒体服务器在网络处理的效率上提 ...

  8. EasyPusher:基于live555的DarwinInjector实现的RTSP直播推送程序

    先简单介绍一下EasyPusher的功能,后面再对具体内部架构做介绍: EasyPusher:https://github.com/EasyDarwin/EasyPusher EasyPusher是什 ...

  9. linux 网络设备,网卡配置 ,相关

    网络设备,网卡配置: Eth0是物理网卡:唯一mac地址,Bcast:广播地址,MAsk:子网掩码, Lo:系统自带的回环的ip地址,可以做一些基本的测试应用,比如没有网卡就用127.0.0.1, r ...

  10. cygwin使用笔记

    1.在cygwin里访问Windows盘 cd /cygdrive/c cd c: 2.整合cygwin命令到Windows中 假设cygwin安装在d:/develop/cygwin,则将d:/de ...