码迷,mamicode.com
首页 > 其他好文 > 详细

【POJ 1741】 Tree

时间:2018-07-15 16:08:29      阅读:172      评论:0      收藏:0      [点我收藏+]

标签:als   oca   深度优先   using   lis   超过   nbsp   queue   define   

【题目链接】

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

【算法】

           点分治

           要求距离不超过k的点对个数,不妨将路径分成两类 :

           1. 经过根节点 2. 不经过根节点

           考虑第1类路径,不妨从根节点进行一次深度优先遍历,求出每个点与根节点的距离,将距离排序,然后用两个指针扫描一遍即可,注意要减掉多算的

           然后,递归地计算子节点的第一类路径,即可

          但是这样是会超时的,我们不妨每次选择子树的重心进行计算 

          这样做法的时间复杂度存在上界 : O(Nlog^2N)

【代码】

            

#include <algorithm>  
#include <bitset>  
#include <cctype>  
#include <cerrno>  
#include <clocale>  
#include <cmath>  
#include <complex>  
#include <cstdio>  
#include <cstdlib>  
#include <cstring>  
#include <ctime>  
#include <deque>  
#include <exception>  
#include <fstream>  
#include <functional>  
#include <limits>  
#include <list>  
#include <map>  
#include <iomanip>  
#include <ios>  
#include <iosfwd>  
#include <iostream>  
#include <istream>  
#include <ostream>  
#include <queue>  
#include <set>  
#include <sstream>  
#include <stdexcept>  
#include <streambuf>  
#include <string>  
#include <utility>  
#include <vector>  
#include <cwchar>  
#include <cwctype>  
#include <stack>  
#include <limits.h> 
using namespace std;
#define MAXN 10010

int i,n,k,tot,len,ans,root,u,v,w;
int head[MAXN],weight[MAXN],size[MAXN],d[MAXN],dep[MAXN];
bool visited[MAXN];

struct Edge
{
    int to,w,nxt;
} e[MAXN<<1];

inline void addedge(int u,int v,int w)
{
    tot++;
    e[tot] = (Edge){v,w,head[u]};
    head[u] = tot;
}
inline void get_root(int u,int fa,int total)
{
    int i,v;
    size[u] = 1;
    weight[u] = 0;
    for (i = head[u]; i; i = e[i].nxt)
    {
        v = e[i].to;
        if (v != fa && !visited[v])
        {
            get_root(v,u,total);
            size[u] += size[v];
            weight[u] = max(weight[u],size[v]);
        }
    }    
    weight[u] = max(weight[u],total-size[u]);
    if (weight[u] < weight[root]) root = u;
}
inline void dfs(int u,int fa)
{
    int i,v,w;
    d[++len] = dep[u];
    for (i = head[u]; i; i = e[i].nxt)
    {
        v = e[i].to;
        w = e[i].w;
        if (!visited[v] && v != fa)
        {
            dep[v] = dep[u] + w;
            dfs(v,u);
        }    
    }
}
inline int calc(int u)
{
    int i,j;
    int ret = 0;
    len = 0;
    dfs(u,0);
    i = 1; j = len;
    sort(d+1,d+len+1);
    while (i < j)
    {
        if (d[i] + d[j] <= k) 
        {
            ret += j - i;
            i++;
        } else j--;
    }
    return ret;
}
inline void work(int u)
{
    int i,v,w;
    dep[u] = 0;
    visited[u] = true;
    ans += calc(u);
    for (i = head[u]; i; i = e[i].nxt)
    {
        v = e[i].to;
        w = e[i].w;
        if (!visited[v])
        {
            dep[v] = w;
            ans -= calc(v);
            root = 0;
            get_root(v,0,size[v]);
            work(root);
        }
    }
}

int main()
{
    
    while (scanf("%d%d",&n,&k) != EOF && !(n == 0 && k == 0))
    {
        memset(visited,false,sizeof(visited));
        tot = 0;
        for (i = 1; i <= n; i++) head[i] = 0;
        for (i = 1; i < n; i++)
        {
            scanf("%d%d%d",&u,&v,&w);
            addedge(u,v,w);
            addedge(v,u,w);
        }    
        size[0] = weight[0] = n;
        root = 0;
        get_root(1,0,0);
        ans = 0;
        work(root);
        printf("%d\n",ans);
    }

    return 0;
}

 

【POJ 1741】 Tree

标签:als   oca   深度优先   using   lis   超过   nbsp   queue   define   

原文地址:https://www.cnblogs.com/evenbao/p/9313628.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!