【cf961G】G. Partitions(组合意义+第二类斯特林数)

传送门

题意:
给出\(n\)个元素,每个元素有价值\(w_i\)。现在要对这\(n\)个元素进行划分,共划分为\(k\)组。每一组的价值为\(|S|\sum_{i=0}^{|S|}w_i\)。
最后询问所有划分的总价值。

思路:
直接枚举划分不好计算,考虑单独计算每一个元素的贡献,那么就有式子:
\[
\sum_{i=1}^nw_i\sum_{j=1}^{n-k+1}{n-1\choose j-1}\begin{Bmatrix}
n - j \\ k - 1
\end{Bmatrix}j
\]
观察这个式子,前面的求和式直接单独计算,后面枚举的上界则可以修改为\(n\)。那么就有:
\[
\begin{aligned}
&\sum_{i=0}^n{n-1\choose i-1}\begin{Bmatrix}
n - i \\ k - 1
\end{Bmatrix}i
\end{aligned}
\]
接下来,就是表演的时刻了。
\[
\begin{aligned}
&\sum_{i=0}^n{n-1\choose i-1}\begin{Bmatrix}
n - i \\ k - 1
\end{Bmatrix}(i-1+1)\=&\sum_{i=0}^n \frac{(n-1)!}{(i-1)!(n-i)!}\begin{Bmatrix}
n - i \\ k - 1
\end{Bmatrix}(i-1)+\sum_{i=0}^n{n-1\choose i-1}\begin{Bmatrix}
n - i \\ k - 1
\end{Bmatrix}\=&(n-1)\sum_{i=0}^n {n-2\choose i-2}\begin{Bmatrix}
n - i \\ k - 1
\end{Bmatrix}+\sum_{i=0}^n{n-1\choose i-1}\begin{Bmatrix}
n - i \\ k - 1
\end{Bmatrix}\=&(n-1)\sum_{i=0}^n {n-2\choose n-i}\begin{Bmatrix}
n - i \\ k - 1
\end{Bmatrix}+\sum_{i=0}^n{n-1\choose n-i}\begin{Bmatrix}
n - i \\ k - 1
\end{Bmatrix}
\end{aligned}
\]
然后我们现在就是要求这样一个式子:
\[
\sum_{i=0}^n{n\choose i}\begin{Bmatrix}
i \\ k - 1
\end{Bmatrix}
\]
我们考虑将第二类斯特林数拆开:
\[
\begin{aligned}
&\sum_{i=0}^n{n\choose i}\begin{Bmatrix}
i \\ k - 1
\end{Bmatrix}\ =&\sum_{i=0}^n{n\choose i}\frac{1}{(k-1)!}\sum_{j=0}^{k-1}(-1)^j{k-1\choose j}(k-1-j)^i\ =&\sum_{j=0}^n\frac{1}{(k-1)!}(-1)^j{k-1\choose j}\sum_{i=0}^n{n\choose i}(k-1-j)^i\ =&\sum_{j=0}^n\frac{1}{(k-1)!}(-1)^j{k-1\choose j}(k-j)^n
\end{aligned}
\]
然后就没了。
整个推导过程第一个关键点是拆\(i\),可能是因为这个\(i\)比较烦,所以将其拆开与前面式子合并。
第二个关键就是利用好组合意义,后面和式的意义就是将元素装入\(k-1-j\)个盒子的方案数,有些元素可能一个盒子都不加入。那么对于每个元素而言,就有\(k-j\)种选择,所以可以直接把和式化简。
然后随便搞一下就行。

还有一种更加精妙的思路,我们还是单独考虑每一个元素,但是当考虑\(i\)这个元素时,若\(j\)和\(i\)在同一个集合中,\(j\)对\(i\)有\(w_i\)的贡献。
那么对一个元素而言分两种情况考虑,一个是自己对自己的贡献,显然方案数有\(\begin{Bmatrix} n \\ k \end{Bmatrix}\);另一个是其它元素对它的贡献,那么考虑将这个元素加入其它元素中(不可能自己单独一个集合),方案数为\((n-1)\begin{Bmatrix} n-1 \\ k \end{Bmatrix}\)。
那么最终一个元素的答案就为:
\[
\begin{Bmatrix}
n \\ k
\end{Bmatrix}+(n-1)\begin{Bmatrix}
n-1 \\ k
\end{Bmatrix}
\]
这个式子随便搞搞就行了。
代码如下(代码为第一种解法):

/*
 * Author:  heyuhhh
 * Created Time:  2019/12/12 17:06:20
 */
#include <iostream>
#include <algorithm>
#include <cstring>
#include <vector>
#include <cmath>
#include <set>
#include <map>
#include <queue>
#include <iomanip>
#define MP make_pair
#define fi first
#define se second
#define sz(x) (int)(x).size()
#define all(x) (x).begin(), (x).end()
#define INF 0x3f3f3f3f
#define Local
#ifdef Local
  #define dbg(args...) do { cout << #args << " -> "; err(args); } while (0)
  void err() { std::cout << '\n'; }
  template<typename T, typename...Args>
  void err(T a, Args...args) { std::cout << a << ' '; err(args...); }
#else
  #define dbg(...)
#endif
void pt() {std::cout << '\n'; }
template<typename T, typename...Args>
void pt(T a, Args...args) {std::cout << a << ' '; pt(args...); }
using namespace std;
typedef long long ll;
typedef pair<int, int> pii;
//head
const int N = 2e5 + 5, MOD = 1e9 + 7;

int n, k;
int fac[N], inv[N];
ll qpow(ll a, ll b) {
    ll ans = 1;
    while(b) {
        if(b & 1) ans = ans * a % MOD;
        a = a * a % MOD;
        b >>= 1;
    }
    return ans;
}
int C(int n, int m) {
    return 1ll * fac[n] * inv[m] % MOD * inv[n - m] % MOD;
}
void init() {
    fac[0] = 1;
    for(int i = 1; i < N; i++) fac[i] = 1ll * fac[i - 1] * i % MOD;
    inv[N - 1] = qpow(fac[N - 1], MOD - 2);
    for(int i = N - 2; i >= 0; i--) inv[i] = 1ll * inv[i + 1] * (i + 1) % MOD;
}

void run(){
    init();
    int sum = 0, ans = 0;
    for(int i = 1; i <= n; i++) {
        int w; cin >> w;
        sum = (sum + w) % MOD;
    }
    if(n == 1) {
        cout << sum << '\n';
        return;
    }
    for(int i = 0, d = 1; i < k; i++, d = MOD - d) {
        int res = 1ll * d * inv[i] % MOD * inv[k - 1 - i] % MOD * qpow(k - i, n - 2) % MOD * (k - i + n - 1) % MOD;
        ans = (ans + res) % MOD;
    }
    ans = 1ll * ans * sum % MOD;
    cout << ans << '\n';
}

int main() {
    ios::sync_with_stdio(false);
    cin.tie(0); cout.tie(0);;
    cout << fixed << setprecision(20);
    while(cin >> n >> k) run();
    return 0;
}

原文地址:https://www.cnblogs.com/heyuhhh/p/12053674.html

时间: 2024-10-07 18:32:56

【cf961G】G. Partitions(组合意义+第二类斯特林数)的相关文章

【CF961G】Partitions 第二类斯特林数

[CF961G]Partitions 题意:给出n个物品,每个物品有一个权值$w_i$,定义一个集合$S$的权值为$W(S)=|S|\sum\limits_{x\in S} w_x$,定义一个划分的权值为$V(R)=\sum\limits_{S\in R} W(S)$.求将n个物品划分成k个集合的所有方案的权值和. $n,k\le 2\cdot 10^5,w_i\le 10^9$ 题解:第二类斯特林数针是太好用辣! 显然每个物品都是独立的,所以我们只需要处理出每个物品被统计的次数即可,说白了就是

poj 3088 组合计数,第二类斯特林数

题意:给出n个数字[1,n],问你可以组成多少种不同的序列,其中每一个序列由几个部分组成,每个部分包含几个数字,每部分内数字无序,部分之间数字有序.每个数字最多使用一次,可以不用. 思路:枚举从n个数字中选出i个数字(组合数),再枚举将这i个数字分成j个部分(第二类斯特林数),然后乘上j的全排列. 1 #include <iostream> 2 #include <cstring> 3 #include <cstdio> 4 using namespace std; 5

关于第二类斯特林数的一丢丢东西

关于第二类斯特林数的一丢丢东西 第二类斯特林数 S(n,m)表示有\(n\)个有区别小球,要放进\(m\)个相同盒子里,且每个盒子非空的方案数 考虑一个很容易的递推: \[S(n,m)=S(n-1,m-1)+m*S(n-1,m)\] 考虑组合意义: 假设前面的\(n-1\)个球丢进了\(m-1\)个组,因为每个组非空,所以这个球只有一种选择--自己一组 如果前面的球已经分成了\(m\)组,那么,这个球就有\(m\)种放法 所以这个递推式就是这样来的 那么,只考虑组合意义可不可以算? 当然是可以的

【BZOJ5093】图的价值(第二类斯特林数,组合数学,NTT)

[BZOJ5093]图的价值(第二类斯特林数,组合数学,NTT) 题面 BZOJ 题解 单独考虑每一个点的贡献: 因为不知道它连了几条边,所以枚举一下 \[\sum_{i=0}^{n-1}C_{n-1}^i·i^k·2^{\frac{n(n-1)}{2}}\] 因为有\(n\)个点,所以还要乘以一个\(n\) 所以,我们真正要求的就是: \[\sum_{i=0}^{n-1}C_{n-1}^i·i^k\] 怎么做? 看到了\(i^k\)想到了第二类斯特林数 \[m^n=\sum_{i=0}^{m}

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(

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

Gym Gym 101147G 第二类斯特林数

题目链接:http://codeforces.com/gym/101147/problem/G 题意:n个人,去参加k个游戏,k个游戏必须非空,有多少种放法? 分析: 第二类斯特林数,划分好k个集合后乘以阶乘: 1 #include <bits/stdc++.h> 2 3 using namespace std; 4 5 const int maxn = 1010; 6 const long long MOD = 1000000000 + 7L; 7 long long stir[maxn][

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

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

题目 在2016年,佳媛姐姐刚刚学习了第二类斯特林数,非常开心. 现在他想计算这样一个函数的值: S(i, j)表示第二类斯特林数,递推公式为: S(i, j) = j ? S(i ? 1, j) + S(i ? 1, j ? 1), 1 <= j <= i ? 1. 边界条件为:S(i, i) = 1(0 <= i), S(i, 0) = 0(1 <= i) 你能帮帮他吗? 输入格式 输入只有一个正整数 输出格式 输出f(n).由于结果会很大,输出f(n)对998244353(7