普通dfs访问每个点对的复杂度是O(n^2)的,显然会超时。

考虑访问到当前子树的根节点时,统计所有经过根的点(u, v)满足:

dist(u) + dist(v) <= maxd,并且

belong(u)≠belong(v)(即u,v不在同一子树)。

这里说的距离指的是节点到跟的距离。

可以用作差法,即用所有满足条件的点对数减去那些在根节点为当前子树根节点的儿子节点的点对数。

上面一步可以用O(nlogn)的复杂度解决,即先排序再比较。

根节点子树可以递归解决,用树的点分治。

总复杂度上界是O(nlognlogn)。

http://poj.org/problem?id=1987

 #include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
const int maxn = 1e5 + ;
const int inf = 0x3f3f3f3f;
struct Edge{
int to, next, w;
}edge[maxn << ];
int head[maxn], N;
int n, m, maxd;
bool vis[maxn];
int cnt[maxn];
int maxi[maxn];
int ans;
int mini, root, sum;
int buf[maxn], k; void addEdge(int u, int v, int w){
edge[N].next = head[u];
edge[N].to = v;
edge[N].w = w;
head[u] = N++;
} void init(){
N = ;
memset(head, -, sizeof head);
} void get_cnt(int u){
cnt[u] = ;
vis[u] = ;
maxi[u] = -;
buf[k++] = u;
for(int i = head[u]; i + ; i = edge[i].next){
int v = edge[i].to;
if(vis[v]) continue;
get_cnt(v);
cnt[u] += cnt[v];
maxi[u] = max(maxi[u], cnt[v]);
}
vis[u] = ;
} void get_dist(int u, int d){
vis[u] = ;
buf[k++] = d;
for(int i = head[u]; i + ; i = edge[i].next){
int v = edge[i].to;
if(vis[v]) continue;
get_dist(v, d + edge[i].w);
}
vis[u] = ;
} int get_buf_sum(int left, int right){
sort(buf + left, buf + right);
int tem = ;
for(int i = right - , j = left; i >= left + ; i--){
while(j < i && buf[i] + buf[j] <= maxd) ++j;
tem += min(i, j) - left;
}
return tem;
} void cal(int u){
k = ;
get_cnt(u);
mini = inf;
for(int i = ; i < k; i++){
int tem = max(cnt[u] - cnt[buf[i]], maxi[buf[i]]);
if(tem < mini) mini = tem, root = buf[i];
}
k = ;
vis[root] = ;
int tem = ;
for(int i = head[root]; i + ; i = edge[i].next){
int v = edge[i].to;
if(vis[v]) continue;
int pre = k;
get_dist(v, edge[i].w);
tem -= get_buf_sum(pre, k);
}
buf[k++] = ;
tem += get_buf_sum(, k);
ans += tem;
for(int i = head[root]; i + ; i = edge[i].next){
int v = edge[i].to;
if(vis[v]) continue;
cal(v);
}
} void solve(){
scanf("%d", &maxd);
memset(vis, , sizeof vis);
ans = ;
for(int i = ; i <= n; i++){
if(!vis[i]) cal(i);
}
printf("%d\n", ans);
} int main(){
//freopen("in.txt", "r", stdin);
while(~scanf("%d", &n)){
scanf("%d", &m);
init();
for(int i = , x, y, z; i < m; i++){
scanf("%d%d%d", &x, &y, &z);
addEdge(x, y, z);
addEdge(y, x, z);
getchar(), getchar();
}
solve();
}
return ;
}

poj1987 Distance Statistics的更多相关文章

  1. 【点分治】poj1741 Tree / poj2114 Boatherds / poj1987 Distance Statistics

    三道题都很类似.给出1741的代码 #include<cstdio> #include<algorithm> #include<cstring> using nam ...

  2. POJ 1987 Distance Statistics 树分治

    Distance Statistics     Description Frustrated at the number of distance queries required to find a ...

  3. BZOJ_3365_[Usaco2004 Feb]Distance Statistics 路程统计&&POJ_1741_Tree_点分治

    BZOJ_3365_[Usaco2004 Feb]Distance Statistics 路程统计&&POJ_1741_Tree_点分治 Description     在得知了自己农 ...

  4. 【poj1987】 Distance Statistics

    http://poj.org/problem?id=1987 (题目链接) 题意 给出一棵树,求树上距离不超过K的点对个数. Solution 点分治,同poj1741. 代码 // poj1987 ...

  5. BZOJ 3365 Distance Statistics 点分治

    这道题是一道点分治的题目,难度不大,可以拿来练手. 关键是对于找出来的重心的删除操作需要删掉这条边,这很重要. 还有每次找重心的时候,不但要考虑他的子节点的siz,还要考虑父节点的siz. 然后就A了 ...

  6. BZOJ 3365: [Usaco2004 Feb]Distance Statistics 路程统计

    Description 一棵树,统计距离不大于 \(k\) 的点对个数. Sol 点分治. 发现自己快把点分治忘干净了... 找重心使所有儿子的最大值尽量小,然后每次处理全部子树,再减去每个子树的贡献 ...

  7. bzoj 3365 [Usaco2004 Feb]Distance Statistics 路程统计(点分治,单调)

    [题意] 求树上长度不超过k的点对数目. [思路] 和 Tree 一样一样的. 就是最后统计的时候别忘把根加上. [代码] #include<set> #include<cmath& ...

  8. POJ 1987 Distance Statistics

    http://poj.org/problem?id=1987 题意:给一棵树,求树上有多少对节点满足距离<=K 思路:点分治,我们考虑把每个距离都存起来,然后排序,一遍扫描计算一下,注意还要减掉 ...

  9. POJ 1987 BZOJ 3365 Distance Statistics 树的分治(点分治)

    题目大意:(同poj1741,刷一赠一系列) CODE: #include <cstdio> #include <cstring> #include <iostream& ...

随机推荐

  1. 根据搜素的字符串改变label包含该字符串的文字

    http://www.2cto.com/kf/201504/391811.html NSString *text =@"人生若只如初见"; //判断字符串所在的位置,并不区分大小写 ...

  2. Java基础(36):String与基本数据类型之间的双向转换(Wrapper类)

    Java 中基本类型和字符串之间的转换 在程序开发中,我们经常需要在基本数据类型和字符串之间进行转换. 其中,基本类型转换为字符串有三种方法: 1. 使用包装类的 toString() 方法 2. 使 ...

  3. 安装Eclipse并配置JacORB插件

    前人成果 •      eclipse中开发corba完整说明(jacORB版) http://blog.csdn.net/hq0927/article/details/8129534 •      ...

  4. demo06

    city_data.xml <?xml version="1.0" encoding="utf-8"?> <resources> < ...

  5. 阿里 drds 分布式数据库分节点查询

    mybatis 模式下,xml 中写法 <select id="selectFailDetailOneNode" resultMap="BaseResultMap& ...

  6. PAT乙级 1013. 数素数 (20)

    1013. 数素数 (20) 时间限制 100 ms 内存限制 65536 kB 代码长度限制 8000 B 判题程序 Standard 作者 CHEN, Yue 令Pi表示第i个素数.现任给两个正整 ...

  7. 为centos添加额外的源

    使用这个命令: yum install epel-release

  8. Ceph的状态错误

    使用命令检查ceph集群的监控状态,得到 [root@node1 ~]# ceph -s cluster c4898b1c-7ac1-406d-bb5d-d3c7980de438 health HEA ...

  9. SessionHelper

    MXS&Vincene  ─╄OvЁ  &0000009 ─╄OvЁ  MXS&Vincene MXS&Vincene  ─╄OvЁ:今天很残酷,明天更残酷,后天很美好 ...

  10. 财务比率:ROE, 净利润增长率、毛利率、市盈率、PEG

    净资产收益率是判断资产回报率指标,是最重要的财务数据 净利润增长率企业成长的参考指标 净利率企业获利能力的指标 毛利率企业获利能力的指标 资产负债率企业偿还债务的能力,也是重要的风险指标之其他还要根据 ...