• bzoj3697 采药人的路径


    3697: 采药人的路径

    Time Limit: 10 Sec  Memory Limit: 128 MB
    Submit: 1540  Solved: 526
    [Submit][Status][Discuss]

    Description

    采药人的药田是一个树状结构,每条路径上都种植着同种药材。
    采药人以自己对药材独到的见解,对每种药材进行了分类。大致分为两类,一种是阴性的,一种是阳性的。
    采药人每天都要进行采药活动。他选择的路径是很有讲究的,他认为阴阳平衡是很重要的,所以他走的一定是两种药材数目相等的路径。采药工作是很辛苦的,所以他希望他选出的路径中有一个可以作为休息站的节点(不包括起点和终点),满足起点到休息站和休息站到终点的路径也是阴阳平衡的。他想知道他一共可以选择多少种不同的路径。

    Input

    第1行包含一个整数N。
    接下来N-1行,每行包含三个整数a_i、b_i和t_i,表示这条路上药材的类型。

    Output

    输出符合采药人要求的路径数目。

    Sample Input

    7
    1 2 0
    3 1 1
    2 4 0
    5 2 0
    6 3 1
    5 7 1

    Sample Output

    1

    HINT

    对于100%的数据,N ≤ 100,000。

    分析:这是一道很难的题,不光是在思维上,代码也有许多地方容易写错.

       将边权变一下:0变成-1,1变成0. 选取一个点当作根,对每个点维护一个点权,表示从这个点到根的边权和. 题目就变成了:求路径,使得这个路径两个端点到路径中的某一个点的边权和为0.点权和对于维护这么一个东西有用.

       路径计数问题通常采用点分治的方法.选取的根恰好就是每次分治的重心. 常规的点分治题的做法是先统计所有经过根的路径,然后统计经过根两端点在同一棵子树内的路径.这种方法适用于能够把数据放在一起处理的题. 这道题显然不行......

       利用树形dp的思想:在解一类树形dp问题的时候,遇到多叉树常常把它看做二叉树. 例如如果处理到第i棵子树,那么可以理解为它有两棵子树:1.前i-1棵子树 2.第i棵子树. 每次先在这两棵子树中查询答案,再把第i棵子树合并到前i-1棵子树中去,这是一种很常见的思路.

       对于这道题而言也是一样的.用一个数组f表示前i棵子树的答案,g表示第i棵子树的答案.统计完答案后就把g累加到f身上. 处理完所有的子树后就清空f,继续找重心dfs下去.

       怎么统计答案呢?考虑4个点,其中u和u'的点权是一样的,v和v'的点权是一样的,u和v的点权是相反数.那么 u'到v,u到v',u到v都是合法的路径. 这里并不关心它们具体的点权是多少,只关注每个点在根节点到它的路径上有没有点权和它一样的点,并且点权是否是相反数.

       如果当前枚举到了第j棵子树,令f[i][0/1]表示前j-1棵子树中,有多少个点的点权是i,并且根到它的路径上没有/有和它点值一样的点. g[i][0/1]则表示当前第j棵子树的.  ans += f[i][1] * g[-i][0] + f[i][0] * g[-i][1] + f[i][1] * g[-i][1].

       一般的f[i][0] * g[-i][0]是不能统计进入答案的.但是当i = 0的时候是可以的. g[0][1]也要统计进入答案,这是由于两个0和根的0在一起就组合成一条合法的路径.这样所有合法的路径讨论完了.

       细节比较多:

       1.-i有可能是负数,怎么办呢? 整体偏移一下就好了. i 变成 maxn + i. 这样的话数组就必须开两倍大. 这里特别容易错.

       2.每次dfs只统计一条路径,记得清空.

       3.每次找重心F数组要清零.

       这道题目主要就是点分治题目的一种比较巧妙的处理方法,分类讨论要考虑清楚所有情况,注意一些细节. 以后需要多打几遍复习复习啊.

    #include <cstdio>
    #include <cmath>
    #include <cstring>
    #include <iostream>
    #include <algorithm>
    
    using namespace std;
    
    typedef long long ll;
    
    const ll maxn = 100010;
    ll n,head[maxn],to[maxn * 2],nextt[maxn * 2],w[maxn * 2],tot = 1;
    ll sizee[maxn],F[maxn],sum,root,f[maxn * 2][2],g[maxn * 2][2],ans;
    ll maxx,maxxx,dep[maxn],flag[maxn * 2],vis[maxn];
    
    void add(ll x,ll y,ll z)
    {
        w[tot] = z;
        to[tot] = y;
        nextt[tot] = head[x];
        head[x] = tot++;
    }
    
    void findroot(ll u,ll fa)
    {
        sizee[u] = 1;
        F[u] = 0;
        for (ll i = head[u]; i; i = nextt[i])
        {
            ll v = to[i];
            if (v == fa || vis[v])
                continue;
            findroot(v,u);
            sizee[u] += sizee[v];
            F[u] = max(F[u],sizee[v]);
        }
        F[u] = max(F[u],sum - sizee[u]);
        if (F[u] < F[root])
            root = u;
    }
    
    void dfs(ll u,ll fa)
    {
        maxx = max(maxx,abs(dep[u]));
        if (flag[dep[u] + maxn])
            g[dep[u] + maxn][1]++;
        else
            g[dep[u] + maxn][0]++;
        flag[dep[u] + maxn]++;
        for (ll i = head[u]; i; i = nextt[i])
        {
            ll v = to[i];
            if (vis[v] || v == fa)
                continue;
            dep[v] = dep[u] + w[i];
            dfs(v,u);
        }
        flag[dep[u] + maxn]--;
    }
    
    void solve(ll u)
    {
        vis[u] = 1;
        maxxx = 0;
        for (ll i = head[u]; i; i = nextt[i])
        {
            ll v = to[i];
            if (vis[v])
                continue;
            dep[v] = w[i];
            maxx = 0;
            dfs(v,u);
            maxxx = max(maxxx,maxx);
            for (ll j = maxn - maxx; j <= maxn + maxx; j++)
            {
                ans += f[2 * maxn - j][0] * g[j][1];
                ans += f[2 * maxn - j][1] * g[j][0];
                ans += f[2 * maxn - j][1] * g[j][1];
            }
            ans += f[maxn][0] * g[maxn][0];
            ans += g[maxn][1];
            for (ll j = maxn - maxx; j <= maxn + maxx; j++)
            {
                f[j][0] += g[j][0];
                f[j][1] += g[j][1];
                g[j][0] = g[j][1] = 0;
            }
        }
        for (ll i = maxn - maxxx; i <= maxn + maxxx; i++)
            f[i][0] = f[i][1] = 0;
        for (ll i = head[u]; i; i = nextt[i])
        {
            ll v = to[i];
            if (vis[v])
                continue;
            sum = F[0] = sizee[v];
            findroot(v,root = 0);
            solve(root);
        }
    }
    
    int main()
    {
        scanf("%lld",&n);
        for (int i = 1; i < n; i++)
        {
            ll x,y,z;
            scanf("%lld%lld%lld",&x,&y,&z);
            z = 2 * z - 1;
            add(x,y,z);
            add(y,x,z);
        }
        F[0] = sum = n;
        findroot(1,0);
        solve(root);
        printf("%lld
    ",ans);
    
        return 0;
    }

       

       

  • 相关阅读:
    发布自己的包到Nuget上
    asp.net core 中的MD5加密
    asp.net core csrf
    KNN算法
    ios测试apk
    python多进程
    机顶盒 gettimeofday()获取毫秒溢出
    Kiggle:Digit Recognizer
    Kaggle:Titanic: Machine Learning from Disaster
    Python抓取微博评论
  • 原文地址:https://www.cnblogs.com/zbtrs/p/8511680.html
Copyright © 2020-2023  润新知