Topcoder SRM 683 Div2 - C

樹形Dp的題,根據題意建樹。ios

DP[i][0] 表示以i爲根節點的樹的包含i的時候的全部狀態點數的總和markdown

Dp[i][1] 表示包含i結點的狀態數目ui

對於一個子節點vspa

Dp[i][0] = (Dp[v][1]+1)*Dp[i][0]+Dp[v][0]*Dp[i][1]code

表示子節點的全部狀態與i的全部的狀態之間的組合(能夠不組合,因此DP[v][1]+1),ip

接下來更新i的狀態數目it

DP[i][1] = Dp[i][1]*(Dp[v][1]+1)io

這樣就能夠算出來i結點爲根結的因此狀態,樹中的全部點的狀態和就是答案。function

#include <vector>
#include <list>
#include <map>
#include <set>
#include <deque>
#include <stack>
#include <bitset>
#include <algorithm>
#include <functional>
#include <numeric>
#include <utility>
#include <sstream>
#include <iostream>
#include <iomanip>
#include <cstdio>
#include <cmath>
#include <cstdlib>
#include <ctime>

using namespace std;

typedef long long LL;

typedef vector<int>VI;

const int Max = 1e5+10;

const LL Mod = 1e9+7;


LL  Dp[Max][2];

LL a[Max];

VI M[Max];

void DFS(int u, int fa)
{
    Dp[u][0] = Dp[u][1] = 1;

    for(int i = 0;i < M[u].size(); i++)
    {   
        if(M[u][i] == fa) continue;

        DFS(M[u][i], u);

        int v = M[u][i];

        Dp[u][0] = (((Dp[v][1] + 1) * Dp[u][0]) % Mod + (Dp[u][1] * Dp[v][0]) % Mod) % Mod;

        Dp[u][1] = (Dp[u][1] * (Dp[v][1]+1)) % Mod;
    }
}

class SubtreesCounting {
public:
    int sumOfSizes(int, int, int, int, int);
};

int SubtreesCounting::sumOfSizes(int n, int a0, int b, int c, int m) 
{
    a[0] = a0;

    for(int i = 1;i <= n-1; i++)
    {
        a[i] = (b * a[i-1] + c) % m;
    }

    for(int i = 1 ;i <= n-1; i++)
    {
        int j = a[i-1] % i;

        M[i].push_back(j);

        M[j].push_back(i);
    }


    LL ans = 0;

    for(int  i = 0; i < n; i++)
    {
        if(!Dp[i][0]) DFS(i,-1);

        ans = (ans + Dp[i][0]) % Mod;
    }

    return ans;
}
相關文章
相關標籤/搜索