设S是一个有限集,A_1,A_2…A_n是S的n个子集,则
∣ S − ⋃ i = 1 n A i ∣ = ∑ i = 0 n ( − 1 ) i ∑ 1 ≤ j 1 < j 2 . . . < j i ≤ n ∣ ⋂ k = 1 i A j k ∣ |S-\bigcup_{i=1}^{n}A_i|=\sum_{i=0}^{n}(-1)^i\sum_{1\leq j_1< j_2...
∣S−⋃i=1nAi∣=∑i=0n(−1)i∑1≤j1<j2...<ji≤n∣⋂k=1iAjk∣
m件不同的物品,分给n个人,要求每一个人至少分得一件物品,求不同的分配方案数
令 A i A_i Ai表示第i个人没有物品, S S S表示 m m m个物品分给 n n n个人的总方案数
则 ∣ S − ⋃ i = 1 n A i ∣ = ∑ i = 0 n ( − 1 ) i ∑ 1 ≤ j 1 < j 2 . . . < j i ≤ n ∣ ⋂ k = 1 i A j k ∣ |S-\bigcup_{i=1}^{n}A_i|=\sum_{i=0}^{n}(-1)^i\sum_{1\leq j_1< j_2...
= ∑ i = 0 n ( − 1 ) i ( n i ) ( n − i ) m =\sum_{i=0}^{n}(-1)^i\binom{n}{i}(n-i)^m =∑i=0n(−1)i(in)(n−i)m
有 2 n 2n 2n个元素 a 1 , a 2 , . . . , a n a_1,a_2, ...,a_n a1,a2,...,an 和 b 1 , b 2 , . . . , b n b_1,b_2, ...,b_n b1,b2,...,bn,求有多少个它们的全排列,满足任意的
1 ≤ i ≤ n 1 \leq i \leq n 1≤i≤n, a i a_i ai 和 b i b_i bi 都不相邻。
同样的,令 A i A_i Ai表示 a i a_i ai和 b i b_i bi相邻,
则 ∣ S − ⋃ i = 1 n A i ∣ = ∑ i = 0 n ( − 1 ) i ∑ 1 ≤ j 1 < j 2 . . . j i ≤ n ∣ ⋂ k = 1 i A j k ∣ |S-\bigcup_{i=1}^{n}A_i|=\sum_{i=0}^{n}(-1)^i\sum_{1\leq j_1< j_2...j_i \leq n}|\bigcap_{k=1}^{i}A_{j_k}| ∣S−⋃i=1nAi∣=∑i=0n(−1)i∑1≤j1<j2...ji≤n∣⋂k=1iAjk∣
= ∑ i = 0 n ( − 1 ) i ( n i ) ∣ 有 i 对相邻的方案数 ∣ =\sum_{i=0}^{n}(-1)^i\binom{n}{i}|有i对相邻的方案数| =∑i=0n(−1)i(in)∣有i对相邻的方案数∣
= ∑ i = 0 n ( − 1 ) i ( n i ) 2 i ∗ ( 2 n − i ) ! =\sum_{i=0}^{n}(-1)^i\binom{n}{i}2^i*(2n-i)! =∑i=0n(−1)i(in)2i∗(2n−i)!
有了以上基本常识就可以上大招了
大意:
给出一个长度为n的序列 a 1 , a 2 . . . a n a_1,a_2...a_n a1,a2...an。求从中选择一个非空子集使得他们的按位与之和等于0的方案数
思路:
不考虑复杂度的话我们有一个非常套路的容斥做法。考虑性质Ai表示子集与之后第i位为1,那么我们的答案其实就是
∣ Ω − A 1 ⋃ A 2 . . . ⋃ A 20 ∣ = ∑ i = 0 20 ( − 1 ) i ∑ 1 ≤ j 1 < j 2 . . . < j i ≤ 20 ∣ A j 1 ⋃ A j 2 . . . A j i ∣ |\Omega -A_1\bigcup A_2...\bigcup A_{20}|=\sum_{i=0}^{20}(-1)^i\sum_{1\leq j_1 < j_2...
其中||符号就表示集合的大小
显然就可以状压枚举,这样的时间复杂度是 O ( n ∗ 1 e 6 ) O(n*1e6) O(n∗1e6),考虑优化。
注意到对于 ∣ A j 1 ⋃ A j 2 . . . A j i ∣ |A_{j_1}\bigcup A_{j_2}...A_{j_i}| ∣Aj1⋃Aj2...Aji∣,我们记满足对应所有性质的元素的个数为k,则该集合的大小就是 2 k − 1 2^k-1 2k−1,那么什么元素会满足这些性质呢?就是二进制为其超集的元素呗,其价值就是1.
所以我们只要做一遍超集后缀和即可,时间复杂度来到 O ( 20 ∗ 1 e 6 ) O(20*1e6) O(20∗1e6)
code
#include
using namespace std;
#define ll long long
#define endl '\n'
const ll N=1e5+10;
const ll mod=1e9+7;
ll n,cnt=0,a;
ll mas[N];
ll up=20;
ll vis[30];
ll dp[(1<<20)+10];
ll ksm(ll x,ll y)
{
ll ans=1;
while(y)
{
if(y&1) ans=ans*x%mod;
x=x*x%mod;
y>>=1;
}
return ans;
}
ll gt()
{
ll tot=0;
ll fl;
for(int i=1;i<=n;++i)
{
fl=1;
for(int j=0;j<up;++j)
{
if(!vis[j]) continue;
if((mas[i]&(1<<j))==0)
{
fl=0;
break;
}
}
if(fl) tot++;
}
return ((ksm(2,tot)-1)%mod+mod)%mod;
}
void solve()
{
cin>>n;
for(int i=1;i<=n;++i) cin>>a,dp[a]++;
for(int j=0;j<up;++j)
{
for(int i=0;i<(1<<up);++i)
{
if((i&(1<<j))==0) dp[i]+=dp[i^(1<<j)];
}
}
ll ans=0;
for(int s=0;s<(1<<up);++s)
{
cnt=0;
for(int i=0;i<up;++i) if(s&(1<<i)) cnt++;
if(cnt%2) ans=((ans-ksm(2,dp[s])+1)%mod+mod)%mod;
else ans=((ans+ksm(2,dp[s])-1)%mod+mod)%mod;
}
cout<<ans<<endl;
}
int main()
{
ios::sync_with_stdio(0);cin.tie(0);cout.tie(0);
solve();
return 0;
}
思路:
个人认为这道题还是很顶的
题目有两个限制:每个人要得到 c i c_i ci票,每个人不能投给自己组
然后这里就有一个很巧妙的转化:我们可以先将同一个组的人放在一起来看,这样就可以将第一个限制稍微弱化一些,最后我们再想办法处理组内的方案数
那么此时的问题如下:每一个组 i i i总共有 n u m i num_i numi人,总共需要 f i f_i fi票,组内的人不能投给自己组,求合法方案数
考虑生成函数处理
我们用 a i a_i ai来表示第 i i i组的变元。那么对于第 i i i组的每一个人来说,它能投给除了自己组外的任河组,所以它的贡献是 a 1 + a 2 + . . . + a i − 1 + a i + 1 + . . . + a n a_1+a_2+...+a_{i-1}+a_{i+1}+...+a_n a1+a2+...+ai−1+ai+1+...+an,我们可以简单记为 s − a i s-a_i s−ai,其中s就表示 ∑ i = 1 n a i \sum_{i=1}^{n}a_i ∑i=1nai
那么我们最后得到式子为 ( s − a 1 ) n u m 1 ( s − a 2 ) n u m 2 . . . ( s − a n ) n u m n (s-a_1)^{num_1}(s-a_2)^{num_2}...(s-a_n)^{num_n} (s−a1)num1(s−a2)num2...(s−an)numn,记为 S S S,而我们的答案显然就是 [ a 1 f 1 a 2 f 2 . . . a n f n ] ( S ) [a_1^{f_1}a_2^{f_2}...a_n^{f_n}](S) [a1f1a2f2...anfn](S).此时不难发现,对于 a i a_i ai的指数,每一个s都可以提供1,而 ( s − a i ) n u m i (s-a_i)^{num_i} (s−ai)numi中的 − a i -a_i −ai也可以提供不多于 n u m i num_i numi的指数。
所以我们考虑每一个 a i a_i ai的次数 det i ( d e t i ≤ n u m i , d e t i ≤ f i ) \det_i(det_i \leq num_i,det_i \leq f_i) deti(deti≤numi,deti≤fi),则每一个i有一个从 n u m i num_i numi中选择 d e t i det_i deti的方案数 ( n u m i d e t i ) \binom{num_i}{det_i} (detinumi),并且 d e t i det_i deti会提供 ( − 1 ) d e t i (-1)^{det_i} (−1)deti的系数,然后剩下的所有指数都由 s s s提供,总共是 ∑ f i − ∑ d e t i = n − ∑ d e t i \sum f_i-\sum det_i=n-\sum det_i ∑fi−∑deti=n−∑deti,要分成若干堆,第i堆有 f i − d e t i f_i-det_i fi−deti个,所以其实是一个可重集,不同组之间是相乘的关系,那么最终的答案其实就是
a n s = ∑ d e t i ≤ f i ( − 1 ) ∑ d e t i ∏ ( ( n u m i d e t i ) ) ( ( n − ∑ d e t i ) ! ( f 1 − d e t 1 ) ! ( f 2 − d e t 2 ) ! . . . ( f n − d e t n ) ! ) ans=\sum_{det_i\leq f_i}(-1)^{\sum det_i}\prod (\binom{num_i}{det_i})\binom{(n-\sum det_i)!}{(f_1-det_1)!(f_2-det_2)!...(f_n-det_n)!} ans=deti≤fi∑(−1)∑deti∏((detinumi))((f1−det1)!(f2−det2)!...(fn−detn)!(n−∑deti)!)
= ∏ n u m i ∗ ∑ d e t i ≤ f i ( − 1 ) ∑ d e t i ( n − ∑ d e t i ) ! ∏ d e t i ! ( n u m i − d e t i ) ! ( f i − d e t i ) ! =\prod num_i *\sum_{det_i\leq f_i}(-1)^{\sum det_i}\frac{(n-\sum det_i)!}{\prod det_i!(num_i-det_i)!(f_i-det_i)!} =∏numi∗deti≤fi∑(−1)∑deti∏deti!(numi−deti)!(fi−deti)!(n−∑deti)!
那么其实n只有200,所以我们可以比较轻松地来得到这个式子的结果
考虑枚举 ∑ d e t i \sum det_i ∑deti,我们记为 d d d,那么再记一个 d p i , j dp_{i,j} dpi,j表示当前处理到了第i组, ∑ x ≤ i d e t i = j \sum_{x \leq i}det_i=j ∑x≤ideti=j,显然只要在过程中枚举合法的 d e t i det_i deti就能完成这个dp了,外层枚举d,内层枚举i,j,最内层枚举 d e t i det_i deti,乍一看时间复杂度好像是 O ( n 4 ) O(n^4) O(n4),但是因为 d e t i ≤ n u m i det_i \leq num_i deti≤numi,而 ∑ n u m i = n \sum num_i=n ∑numi=n,所以实际复杂度只有 O ( n 3 ) O(n^3) O(n3)
这样我们就完成了组与组之间的关系。考虑组内部的票数分配, n u m i num_i numi个人,每一个人需要 c i c_i ci票,总共 ∑ c i = f i \sum c_i=f_i ∑ci=fi票,所以这其实还是一个多重集,那么我们只要乘上系数 f i ! ∏ c i ! \frac{f_i!}{\prod c_i!} ∏ci!fi!即可
推推式子还是有点累的~
code
#include
using namespace std;
#define ll long long
#define endl '\n'
const ll N=210;
const ll mod=998244353;
ll n;
ll f[N],t[N],c[N],num[N];//每一组的总期望票数,组别,个人期望票数,每组人数
ll dp[N][N];
ll ksm(ll x,ll y)
{
ll ans=1;
while(y)
{
if(y&1) ans=ans*x%mod;
x=x*x%mod;
y>>=1;
}
return ans;
}
ll inv(ll x)
{
return ksm(x,mod-2);
}
ll p[N],pp[N];
void init(ll n)
{
p[0]=1;
for(ll i=1;i<=n;++i) p[i]=p[i-1]*i%mod;
pp[n]=inv(p[n]);
for(int i=n-1;i>=0;--i) pp[i]=pp[i+1]*(i+1)%mod;
}
void solve()
{
init(200);
// for(int i=1;i<=10;++i) cout<
cin>>n;
for(int i=1;i<=n;++i) cin>>c[i];//期望票数
for(int i=1;i<=n;++i) cin>>t[i],f[t[i]]+=c[i];
for(int i=1;i<=n;++i) num[t[i]]++;
ll ans=1,sum=0;
for(int i=1;i<=n;++i) ans=ans*p[num[i]]%mod;
for(int d=0;d<=n;++d)//det_i之和
{
for(int i=1;i<=n;++i) for(int j=0;j<=n;++j) dp[i][j]=0;
dp[0][0]=p[n-d];
for(int i=1;i<=n;++i)
{
//当前枚举到第i组
for(int j=0;j<=d;++j)//到第i个人位置det_i的总和为j
{
for(int k=0;k<=min(f[i],num[i])&&k<=j;++k)//det_i=k
{
ll det=pp[k]*pp[num[i]-k]%mod*pp[f[i]-k]%mod;
dp[i][j]=(dp[i][j]+det*dp[i-1][j-k]%mod)%mod;
}
}
}
if((d%2)==0) sum=(sum+ans*dp[n][d]%mod)%mod;
else sum=((sum-ans*dp[n][d]%mod)%mod+mod)%mod;
}
for(int i=1;i<=n;++i)//组内多重集
{
sum=sum*p[f[i]]%mod*pp[c[i]]%mod;
}
cout<<sum<<endl;
}
int main()
{
ios::sync_with_stdio(0);cin.tie(0);cout.tie(0);
solve();
return 0;
}
http://acm.hdu.edu.cn/contest/problem?cid=1102&pid=1011
23HDU 09K Cargo
大意:
n家店,每家店卖一种商品,总共有m种商品在卖。k次操作,每次随机选择一家店并且买下对应种类的商品。对于每一种商品i,记ci为售卖该种商品的店数,如果k次操作手里刚好有ci种i商品,且分别来自不同的店,则该商品合法。问k次操作之后所有种类的商品都不合法的方案数%998244353
n , m ≤ 1 e 5 , k < 998244353 n,m\leq 1e5,k<998244353 n,m≤1e5,k<998244353
思路:
可能数据范围并不是很支持直接容斥,但是可以发现基本上就是容斥的样子,所以试试看优化容斥。
令 A i A_i Ai表示第种商品合法的方案数,
则 ∣ S − ⋃ i = 1 m A i ∣ = ∑ i = 0 m ( − 1 ) i ∑ 1 ≤ j 1 < j 2 . . . j i ∣ ⋂ k = 1 i A j k ∣ |S-\bigcup_{i=1}^{m}A_i|=\sum_{i=0}^{m}(-1)^i\sum_{1\leq j_1< j_2...j_i}|\bigcap_{k=1}^{i}A_{j_k}| ∣S−⋃i=1mAi∣=∑i=0m(−1)i∑1≤j1<j2...ji∣⋂k=1iAjk∣
只考虑 A i A_i Ai,方案数为 A n c i ( n − c i ) k − c i A_{n}^{c_i}(n-c_i)^{k-c_i} Anci(n−ci)k−ci,意义显然。多个 A i A_i Ai取交的话,方案数就是 A n ∑ c i ( n − ∑ c i ) k − ∑ c i A_{n}^{\sum c_i}(n-\sum c_i)^{k-\sum c_i} An∑ci(n−∑ci)k−∑ci
发现这个方案数只与 ∑ c i \sum c_i ∑ci有关,而与顺序,某一个具体值无关,所以我们可以改变一下思路,转为枚举 ∑ c i \sum c_i ∑ci
具体来说,对于每一个 ∑ c i = j \sum c_i=j ∑ci=j,它的贡献就是用偶数个 c i c_i ci累加得到j的方案数减去用奇数个 c i c_i ci累加得到j的方案数,(这一步其实就是容斥原理公式的转化)
既然如此,我们考虑生成函数 1 − x c i 1-x^{c_i} 1−xci,表示对于每一个 c i c_i ci,取一个数得到和为 c i c_i ci的方案数为-1,这是因为1是一个奇数,那么显然,在 ∏ ( 1 − x c i ) \prod (1-x^{c_i}) ∏(1−xci)中,如果某一个指数是由偶数个数累加得到的,其系数自然会是正数,奇数同理。由此该生成函数就满足了我们的条件,我们只要对每一项 [ x j ] ( ∏ ( 1 − x c i ) ) [x_j](\prod (1-x^{c_i})) [xj](∏(1−xci))乘上对应的方案数系数 A n ∑ c i ( n − ∑ c i ) k − ∑ c i A_{n}^{\sum c_i}(n-\sum c_i)^{k-\sum c_i} An∑ci(n−∑ci)k−∑ci即可.
个人感觉这种题目应该还是有点套路可循的,因为如果数据范围小一点的话,就是明显显的容斥,然后找到性质用生成函数来优化就不是那么难想到的了
code
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
using namespace std;
#define N maxn
#define db double
#define il inline
#define fir first
#define sec second
#define eps (1e-8)
#define pb push_back
#define ll long long
#define mkp make_pair
#define eb emplace_back
#define pii pair<int, int>
#define lowbit(a) (a & (-a))
#define SZ(a) ((int)a.size())
#define ull unsigned long long
#define all(a) a.begin(), a.end()
#define split cout << "=========\n";
#define GG { cout << "NO\n"; return; }
#define pll pair<long long, long long>
#define equals(a, b) (fabs((a) - (b)) < eps)
constexpr int ON = 0;
constexpr int CW = -1;
constexpr int CCW = 1;
constexpr int BACK = 2;
constexpr int FRONT = -2;
const db pi = acos(-1.000);
constexpr int maxn = 2e5 + 50;
constexpr int INF = 0x3f3f3f3f;
constexpr ll LINF = 0x3f3f3f3f3f3f3f3f;
constexpr int mod = 998244353; /* 1e9 + 7 */
constexpr int dir[8][2] = {-1, 0, -1, 1, 0, 1, 1, 1, 1, 0, 1, -1, 0, -1, -1, -1};
mt19937_64 rnd(random_device {}());
uniform_int_distribution<ull> dist(0, ULLONG_MAX);//use dist(rnd)
bool BEGIN;
ll qpow(ll x,ll y)
{
ll ans=1;
while(y)
{
if(y&1) ans=ans*x%mod;
x=x*x%mod;
y>>=1;
}
return ans;
}
namespace Poly
{
#define mul(x, y) (1ll * x * y >= mod ? 1ll * x * y % mod : 1ll * x * y)
#define minus(x, y) (1ll * x - y < 0 ? 1ll * x - y + mod : 1ll * x - y)
#define plus(x, y) (1ll * x + y >= mod ? 1ll * x + y - mod : 1ll * x + y)//上面其实没用到
#define ck(x) (x >= mod ? x - mod : x)//取模运算太慢了
typedef vector<int> poly;
const int G = 3;//根据具体的模数而定,原根可不一定不一样!!!
//一般模数的原根为 2 3 5 7 10 6
const int inv_G = qpow(G, mod - 2),tt = 22;
int deer[2][tt][(1 << tt)];
vector<int>RR(1 << (tt + 1), 0),inv(1 << tt, 0);
void init(const int t) {//预处理出来NTT里需要的w和wn,砍掉了一个log的时间
assert(t < tt);//一定要注意!!
for(int p = 1; p <= t; ++ p) {
int buf1 = qpow(G, (mod - 1) / (1 << p));
int buf0 = qpow(inv_G, (mod - 1) / (1 << p));
deer[0][p][0] = deer[1][p][0] = 1;
for(int i = 1; i < (1 << p); ++ i) {
deer[0][p][i] = 1ll * deer[0][p][i - 1] * buf0 % mod;//逆
deer[1][p][i] = 1ll * deer[1][p][i - 1] * buf1 % mod;
}
}
inv[1] = 1;
for(int i = 2; i <= (1 << t); ++ i)
inv[i] = 1ll * inv[mod % i] * (mod - mod / i) % mod;
}
int NTT_init(int n) {//快速数论变换预处理
int limit = 1, L = 0;
while(limit <= n) limit <<= 1, L ++ ;
assert(L < tt);
assert(limit < 1 << (tt + 1));
for(int i = 0; i < limit; ++ i)
RR[i] = (RR[i >> 1] >> 1) | ((i & 1) << (L - 1));
return limit;
}
void NTT(poly &A, bool type, int limit) {//快速数论变换
A.resize(limit);
for(int i = 0; i < limit; ++ i)
if(i < RR[i])
swap(A[i], A[RR[i]]);
for(int mid = 2, j = 1; mid <= limit; mid <<= 1, ++ j) {
int len = mid >> 1;
for(int pos = 0; pos < limit; pos += mid) {
// auto wn = deer[type][j].begin();
for(int i = pos, p = 0; i < pos + len; ++ i, ++ p) {
int tmp = 1ll * deer[type][j][p] * A[i + len] % mod;
A[i + len] = ck(A[i] - tmp + mod);
A[i] = ck(A[i] + tmp);
}
}
}
if(type == 0) {
for(int i = 0; i < limit; ++ i)
A[i] = 1ll * A[i] * inv[limit] % mod;
}
}
poly poly_mul(poly A, poly B) {//多项式乘法
int deg = A.size() + B.size() - 1;
int limit = NTT_init(deg);
poly C(limit);
NTT(A, 1, limit);
NTT(B, 1, limit);
for(int i = 0; i < limit; ++ i)
C[i] = 1ll * A[i] * B[i] % mod;
NTT(C, 0, limit);
C.resize(deg);
return C;
}
poly poly_inv(poly &f, int deg) {//多项式求逆 deg
if(deg == 1)
return poly(1, qpow(f[0], mod - 2));
poly A(f.begin(), f.begin() + deg);
poly B = poly_inv(f, (deg + 1) >> 1);
int limit = NTT_init(deg << 1);
NTT(A, 1, limit), NTT(B, 1, limit);
for(int i = 0; i < limit; ++ i)
A[i] = B[i] * (2 - 1ll * A[i] * B[i] % mod + mod) % mod;
NTT(A, 0, limit);
A.resize(deg);
return A;
}
poly poly_dev(poly f) {//多项式求导
int n = f.size();
for(int i = 1; i < n; ++ i) f[i - 1] = 1ll * f[i] * i % mod;
if(n > 1)f.resize(n - 1);
else f[0] = 0;
return f.resize(n - 1), f;//求导整体左移,第0项不要
}
poly poly_idev(poly f) {//多项式求积分
int n = f.size();
for(int i = n - 1; i ; -- i) f[i] = 1ll * f[i - 1] * inv[i] % mod;
return f[0] = 0, f;//积分整体右移,第0项默认为0
}
poly poly_ln(poly f, int deg) {//多项式求对数,第一项为1
poly A = poly_idev(poly_mul(poly_dev(f), poly_inv(f, deg)));
return A.resize(deg), A;
}
poly poly_exp(poly &f, int deg) {//多项式求指数,第一项为0
if(deg == 1)
return poly(1, 1);
poly B = poly_exp(f, (deg + 1) >> 1);
B.resize(deg);
poly lnB = poly_ln(B, deg);
for(int i = 0; i < deg; ++ i)
lnB[i] = ck(f[i] - lnB[i] + mod);
int limit = NTT_init(deg << 1);//n -> n^2
NTT(B, 1, limit), NTT(lnB, 1, limit);
for(int i = 0; i < limit; ++ i)
B[i] = 1ll * B[i] * (1 + lnB[i]) % mod;
NTT(B, 0, limit);
B.resize(deg);
return B;
}
poly poly_pow(poly f, int k) {//多项式快速幂,第一项得是1
f = poly_ln(f, f.size());
for(auto &x : f) x = 1ll * x * k % mod;
return poly_exp(f, f.size());
}
poly poly_ksm(poly f, int k,int m) {//多项式快速幂,适用于初始只有几项,同时所有项都需要的情况,会比上面那个快一点
poly res(1, 1);
while(k){
if(k & 1)
{
res = poly_mul(res, f);
res.resize(m,0);
}
f = poly_mul(f, f);
f.resize(m,0);
k >>= 1;
}
return res;
}
}
using Poly::poly;
using Poly::poly_pow;
using Poly::poly_ksm;
using Poly::poly_mul;
using Poly::poly_inv;
vector<ll> vt;
poly poly_f(int l,int r,int num)
{
if(l>r)
{
return poly(1,1);
}
if(l==r)
{
poly ans(vt[l]+1,0);
ans[0]=1;
ans[vt[l]]=(mod-1)%mod;
return ans;
}
int mid=(l+r)>>1;
poly f = poly_f(l,mid,num);
poly g = poly_f(mid+1,r,num);
f = poly_mul(f,g);
// f.resize(num+1);
return f;
}
ll p[N],pp[N],C[N];
void init(ll n)
{
p[0]=1;
for(ll i=1;i<=n;++i) p[i]=p[i-1]*i%mod;
pp[1]=1;
for(ll i=2;i<=n;++i)
{
pp[i]=(mod-mod/i)*pp[mod%i]%mod;
}
// for(ll i=1;i<=n;++i) C[i]=C[i-1]*qpow(i,mod-2)%mod*(k-i+1)%mod;
}
void init2(ll n,ll k)
{
C[0]=1;
for(ll i=1;i<=n;++i) C[i]=C[i-1]*pp[i]%mod*(k-i+1)%mod;
for(int i=1;i<=n;++i) C[i]=C[i]*p[i]%mod;
}
ll n,m,k;
map<ll,ll> num1;
void solve()
{
num1.clear();
cin>>n>>m>>k;
init2(max(n,m),k);
// for(int i=0;i<=10;++i) cout<
// cout<
for(int i=1;i<=n;++i)
{
ll a;cin>>a;
num1[a]++;
}
vt.clear();
for(auto i:num1) vt.push_back(i.second);
Poly::init(20);
poly f=poly_f(0,vt.size()-1,n);
ll ans=0;
for(ll i=0;i<=min(n,k);++i)
{
// cout<
ans=((ans+f[i]*C[i]%mod*qpow(n-i,k-i)%mod)%mod+mod)%mod;
}
cout<<ans*qpow(qpow(n,k),mod-2)%mod<<endl;
}
bool END;
signed main() {
// cout << fixed << setprecision(10);
ios::sync_with_stdio(false); cin.tie(nullptr);
init(200000);
int T; cin >> T; while (T--)
solve();
// cout << ((&END - & BEGIN) >> 21) << '\n';
return 0;
}
未完待续