BZOJ 2159: Crash 的文明世界(树形dp+第二类斯特林数+组合数)

  • 题意:

给定一棵 \(n\) 个点的树和一个常数 \(k\) , 对于每个 \(i\) , 求

\[\displaystyle S(i) = \sum _{j=1} ^ {n} \mathrm{dist}(i, j)^k\]

\(n ≤ 50000, k ≤ 150\)

  • 题解 :

先划划那个 \(S(i)\) 的式子

我们需要知道一个化 \(x^n(n \ge 0)\) 的东西qwq

\[\displaystyle x^n=\sum_{k=0}^{n}\begin{Bmatrix} n \\ k \end{Bmatrix} x^{\underline k}=\sum _{k=0}^{n}(-1)^k \begin{Bmatrix} n \\ k \end{Bmatrix} {\overline x}\]

这个式子十分的有用,可以转化很多幂指数的东西为斯特林数。

\[\displaystyle S(i)=\sum _{j=1}^{n}\sum_{l=0}^{k}\begin{Bmatrix} k \\ l \end{Bmatrix} \mathrm{dist}(i,j)^{\underline l}\]

然后换个位置

\[\displaystyle S(i)=\sum_{l=0}^{k}\begin{Bmatrix} k \\ l \end{Bmatrix}\sum _{j=1}^{n} \mathrm{dist}(i,j)^{\underline l}\]

然后用一下组合数的一个定义式子:

\[\displaystyle \binom n k = \frac{n!}{(n-k)!k!}=\frac{n^{\underline k}}{k!}\]

\[\therefore \displaystyle n^{\underline k}=\binom n k k!\]

这也可以导出下降幂了

\[\displaystyle S(i)=\sum_{l=0}^{k}\begin{Bmatrix} k \\ l \end{Bmatrix} l!\sum _{j=1}^{n} \binom {\mathrm{dist}(i,j)} l\]

前面那一部分显然是稳定不变的,我们就可以去维护第二部分啦

令 \[\displaystyle dp[i][l]=\sum _{j=1}^{n} \binom {\mathrm{dist}(i,j)} l\]

由于是组合数我们就可以直接套用它的一个递推式来转移了(因为转移的时候,所有 \(\mathrm{dist}(i,j)\) 同增减 \(1\) )

\[\displaystyle \binom n k = \binom {n-1} {k} + \binom {n-1} {k-1}\]

同样的,就有 \(dp[i][l]=dp[j][l]+dp[j][l-1]\) 此处 \(j\) 是 \(i\) 的一个儿子。(这个递推式用来转移真的是巧妙啊qwq)

然后我们要算两个 \(dp\) 值,一个 \(f_{i,l}\) 统计子树的,一个 \(g_{i,l}\) 统计子树外的。

统计子树外的时候,要先算父亲那过来的贡献,然后再算兄弟的贡献。

算兄弟的贡献可以用父亲贡献减掉自己的贡献(见代码中分步写的 \(g_{i,j}\) 的转移) 而且要先转移,再遍历

所以最后 \(O(nk)\) 个状态, \(O(1)\) 的转移,总复杂度就是 \(\Theta(nk)\) .

那个解压输入直接拷贝了 Hany01 大佬的 qwq不会写

代码:

/**************************************************************
    Problem: 2159
    User: zjp_shadow
    Language: C++
    Result: Accepted
    Time:4156 ms
    Memory:67680 kb
****************************************************************/

#include <bits/stdc++.h>
#define For(i, l, r) for(register int i = (l), i##end = (int)(r); i <= i##end; ++i)
#define Fordown(i, r, l) for(register int i = (r), i##end = (int)(l); i >= i##end; --i)
#define Set(a, v) memset(a, v, sizeof(a))
using namespace std;

inline bool chkmin(int &a, int b) {return b < a ? a = b, 1 : 0;}
inline bool chkmax(int &a, int b) {return b > a ? a = b, 1 : 0;}

inline int read() {
    int x = 0, fh = 1; char ch = getchar();
    for (; !isdigit(ch); ch = getchar()) if (ch == ‘-‘) fh = -1;
    for (; isdigit(ch); ch = getchar()) x = (x * 10) + (ch ^ 48);
    return x * fh;
}

void File() {
#ifdef zjp_shadow
    freopen ("2159.in", "r", stdin);
    freopen ("2159.out", "w", stdout);
#endif
}

const int Mod = 10007, N = 50010;

vector<int> G[N];

int n, k, S[160][160];
int fac[160];

void Init(int maxn) {
    S[0][0] = 1; For (i, 1, maxn) { S[i][1] = 1; For (j, 1, i) S[i][j] = (j * S[i - 1][j] % Mod + S[i - 1][j - 1]) % Mod; }
    fac[0] = fac[1] = 1; For (i, 2, maxn) fac[i] = fac[i - 1] * i % Mod;
}

int f[N][160], sz[N];

void Dfs1(int u, int fa) {
    f[u][0] = 1; sz[u] = 1;
    For (i, 0, G[u].size() - 1) {
        int v = G[u][i]; if (v == fa) continue ;
        Dfs1(v, u); sz[u] += sz[v];
        (f[u][0] += f[v][0]) %= Mod;
        For (j, 1, k) (f[u][j] += f[v][j] + f[v][j - 1]) %= Mod;
    }
}

int g[N][160];

void Dfs2(int u, int fa) {
    g[u][0] = (n - sz[u]) % Mod;
    if (fa) {
        For (i, 1, k) {
            g[u][i] = g[fa][i] + g[fa][i - 1];
            g[u][i] += f[fa][i] - (f[u][i] + f[u][i - 1]);
            g[u][i] += f[fa][i - 1] - (f[u][i - 1] + (i > 1 ? f[u][i - 2] : 0));
            g[u][i] = (g[u][i] % Mod + Mod) % Mod;
        }
    }
    For (i, 0, G[u].size() - 1) { int v = G[u][i]; if (v == fa) continue ; Dfs2(v, u); }
}

int ans[N];

inline void Input_Umcompress()
{
    register int l, now, a, b, q, tmp, u, v;
    n = read(), k = read(), l = read(), now = read(), a = read(), b = read(), q = read();
    For(i, 1, n - 1)
        now = (now * a + b) % q, tmp = i < l ? i : l,
        u = i - now % tmp, v = i + 1, G[u].push_back(v), G[v].push_back(u);
}

int main () {
    File(); Init(150);
    Input_Umcompress();
    /*n = read(); k = read();
    For (i, 1, n - 1) {
        int u = read(), v = read();
        G[u].push_back(v);
        G[v].push_back(u);
    }*/

    Dfs1(1, 0); Dfs2(1, 0);

    For (i, 1, n) {
        For (l, 0, k)
            (ans[i] += S[k][l] * fac[l] % Mod * (f[i][l] + g[i][l]) % Mod) %= Mod;
        printf ("%d\n", ans[i]);
    }

    return 0;
}

原文地址:https://www.cnblogs.com/zjp-shadow/p/8641428.html

时间: 2024-07-28 17:53:09

BZOJ 2159: Crash 的文明世界(树形dp+第二类斯特林数+组合数)的相关文章

bzoj 2159: Crash 的文明世界

2159: Crash 的文明世界 Time Limit: 10 Sec  Memory Limit: 259 MBSubmit: 478  Solved: 233[Submit][Status][Discuss] Description Crash 小朋友最近迷上了一款游戏——文明5(Civilization V).在这个游戏中,玩家可以建立和发展自己的国家,通过外交和别的国家交流,或是通过战争征服别的国家.现在Crash 已经拥有了一个N 个城市的国家,这些城市之间通过道路相连.由于建设道路

bzoj 2159 Crash 的文明世界 —— 第二类斯特林数+树形DP

题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2159 使用公式:\( n^{k} = \sum\limits_{i=0}^{k} S(k,i) * i! * C_{n}^{i} \) 所以维护 \( f[x][i] = \sum\limits_{u\in subtree[x],d=dist(x,u)}^{n} C_{d}^{i} \) 然后利用 \( C_{n}^{m} = C_{n-1}^{m} + C_{n-1}^{m-1} \),

bzoj 5093 [Lydsy1711月赛]图的价值 NTT+第二类斯特林数

[Lydsy1711月赛]图的价值 Time Limit: 30 Sec  Memory Limit: 256 MBSubmit: 245  Solved: 128[Submit][Status][Discuss] Description “简单无向图”是指无重边.无自环的无向图(不一定连通). 一个带标号的图的价值定义为每个点度数的k次方的和. 给定n和k,请计算所有n个点的带标号的简单无向图的价值之和. 因为答案很大,请对998244353取模输出. Input 第一行包含两个正整数n,k(

P4827 [国家集训队] Crash 的文明世界(第二类斯特林数+树形dp)

传送门 对于点\(u\),所求为\[\sum_{i=1}^ndis(i,u)^k\] 把后面那堆东西化成第二类斯特林数,有\[\sum_{i=1}^n\sum_{j=0}^kS(k,j)\times j!\times{dis(i,u)\choose j}\] \[\sum_{j=1}^nS(k,j)\times j!\sum_{i=0}^k{dis(i,u)\choose j}\] 于是对于每个点只要维护好\(\sum_{i=0}^k{dis(i,u)\choose j}\)就好了 因为\({n

HDU 4045 Machine scheduling (第二类斯特林数+DP)

A Baidu's engineer needs to analyze and process large amount of data on machines every day. The machines are labeled from 1 to n. On each day, the engineer chooses r machines to process data. He allocates the r machines to no more than m groups ,and

HDU-4625 JZPTREE (树上dp,第二类斯特林数)

题目链接:HDU-4625 JZPTREE 题意 给出$n$个结点的一棵树,对每一个点$x$求所有点到$x$的距离的$k$次方之和.$1\leq n\leq 50000, 1\leq k\leq 500$. 思路 用$Tree_x$表示这棵树以$x$为根,$f(x,k)$表示所有点到$x$的距离的$k$次方之和,$dis(x,y)$表示结点$x$和$y$之间的距离,题意即求:$$f(x,k) = \sum_{y\in Tree_x}{dis(x,y)^k} \tag{1}$$我们要将之转换为能在

【BZOJ 4555】[Tjoi2016&amp;Heoi2016]求和 NTT+第二类斯特林数

用到第二类斯特林数的性质,做法好像很多,我打的是直接ntt,由第二类斯特林数的容斥公式可以推出,我们可以对于每一个i,来一次ntt求出他与所有j组成的第二类斯特林数的值,这个时候我们是O(n^2logn)的,还不如暴力,但是我们发现,对于刚刚提到的容斥的式子,将其化为卷积形式后,其一边的每一项对于每一个i都相同,另一边的每一项是对于所有的i形成一个n项的等比数列,这样我们可以把成等比数列的一边求和,用固定的一边去卷他们的和,这时候的答案的每一项就是所有的i的这一项的和,然后我们再O(n)乘上阶乘

bzoj 4555 [Tjoi2016&amp;Heoi2016]求和 NTT 第二类斯特林数 等比数列求和优化

[Tjoi2016&Heoi2016]求和 Time Limit: 40 Sec  Memory Limit: 128 MBSubmit: 679  Solved: 534[Submit][Status][Discuss] Description 在2016年,佳媛姐姐刚刚学习了第二类斯特林数,非常开心. 现在他想计算这样一个函数的值: S(i, j)表示第二类斯特林数,递推公式为: S(i, j) = j ∗ S(i − 1, j) + S(i − 1, j − 1), 1 <= j &l

bzoj 5093 图的价值 —— 第二类斯特林数+NTT

题目:https://www.lydsy.com/JudgeOnline/problem.php?id=5093 每个点都是等价的,从点的贡献来看,得到式子: \( ans = n * \sum\limits_{d=0}^{n-1} d^{k} * 2^{C_{n-1}^{2}} * C_{n-1}^{d} \) 使用 \( n^{k} = \sum\limits_{i=0}^{k} S(k,i) * i! *C_{n}^{i} \) 得到 \( ans = n * \sum\limits_{d