【LG4491】[HAOI2018]染色

题面

洛谷

题解

颜色的数量不超过\(lim=min(m,\frac nS)\)

考虑容斥,计算恰好出现\(S\)次的颜色至少\(i\)种的方案数\(f[i]\),钦定\(i\)种颜色至少放\(S\)种

有\(m\)种颜色,那么要乘上\(C_m^i\)。

然后这\(n\)个位置分为\(i+1\)个部分:被钦定的\(i\)种颜色,每个\(S\)个;剩下\(m-i\)种颜色,一共\(n-iS\)种颜色,可以看作可重的全排列数,那么就有\(\frac{n!}{(S!)^i(n-iS)!}\),但是后面情况的每个部分有\(m-i\)种取法,所以还要乘上\((m-i)^{n-iS}\)

\(\therefore f[i]=C_m^i\frac{n!}{(S!)^i(n-iS)!}(m-i)^{n-iS}\)

那么答案是什么呢?

\(ans[i]:\)恰好出现\(S\)次的颜色恰好\(i\)种的方案数

用容斥搞一下:

\[ans[i]=\sum_{j=i}^{lim}(-1)^{j-i}C_j^if[j]\\
=\sum_{j=i}^{lim}(-1)^{j-i}\frac{j!}{i!(j-i)!}f[j]\\
\Leftrightarrow
ans[i]*i!=\sum_{j=i}^{lim}\frac{(-1)^{j-i}}{(j-i)!}f[j]*j!
\]

然后\(NTT\)就可以了。

代码

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <algorithm>
using namespace std;
inline int gi() {
register int data = 0, w = 1;
register char ch = 0;
while (!isdigit(ch) && ch != '-') ch = getchar();
if (ch == '-') w = -1, ch = getchar();
while (isdigit(ch)) data = 10 * data + ch - '0', ch = getchar();
return w * data;
}
const int Mod = 1004535809;
int fpow(int x, int y) {
int res = 1;
while (y) {
if (y & 1) res = 1ll * res * x % Mod;
x = 1ll * x * x % Mod;
y >>= 1;
}
return res;
}
const int G = 3, iG = fpow(G, Mod - 2);
const int MAX_N = 1e7 + 5, MAX_M = 1e5 + 5;
int fac[MAX_N];
int C(int n, int m) {
if (m > n) return 0;
return 1ll * fac[n] * fpow(fac[n - m], Mod - 2) % Mod * fpow(fac[m], Mod - 2) % Mod;
}
int N, M, S, Limit, mx;
int W[MAX_N], A[MAX_N << 2], B[MAX_N << 2], rev[MAX_N << 2];
void NTT(int *p, int op) {
for (int i = 0; i < Limit; i++) if (rev[i] > i) swap(p[rev[i]], p[i]);
for (int i = 1; i < Limit; i <<= 1) {
int rot = fpow(op == 1 ? G : iG, (Mod - 1) / (i << 1));
for (int j = 0, pls = (i << 1); j < Limit; j += pls) {
int w = 1;
for (int k = 0; k < i; k++, w = 1ll * w * rot % Mod) {
int x = p[j + k], y = 1ll * w * p[i + j + k] % Mod;
p[j + k] = (x + y) % Mod, p[i + j + k] = (x - y + Mod) % Mod;
}
}
}
}
int main () {
N = gi(), M = gi(), S = gi();
for (int i = 0; i <= M; i++) W[i] = gi();
mx = max(N, M);
fac[0] = 1; for (int i = 1; i <= mx; i++) fac[i] = 1ll * fac[i - 1] * i % Mod;
int mn = min(M, N / S);
Limit = 1; int P = 0;
while (Limit < (mn + 1) << 1) Limit <<= 1, ++P;
for (int i = 0; i <= mn; i++)
A[i] = 1ll * fac[i] * C(M, i) % Mod *
fac[N] % Mod * fpow(M - i, N - i * S) % Mod *
fpow(1ll * fpow(fac[S], i) * fac[N - i * S] % Mod, Mod - 2) % Mod;
for (int i = 0; i <= mn; i++) {
B[i] = fpow(fac[mn - i], Mod - 2);
if ((mn - i) & 1) B[i] = Mod - B[i];
}
for (int i = 0; i < Limit; i++) rev[i] = (rev[i >> 1] >> 1) | ((i & 1) << (P - 1));
NTT(A, 1), NTT(B, 1);
for (int i = 0; i < Limit; i++) A[i] = 1ll * A[i] * B[i] % Mod;
NTT(A, 0);
int inv = fpow(Limit, Mod - 2);
for (int i = 0; i < Limit; i++) A[i] = 1ll * inv * A[i] % Mod;
int ans = 0;
for (int i = 0; i <= mn; i++) ans = (ans + 1ll * W[i] * A[mn + i] % Mod * fpow(fac[i], Mod - 2) % Mod) % Mod;
printf("%d\n", ans);
return 0;
}

【LG4491】[HAOI2018]染色的更多相关文章

  1. BZOJ 5306 [HAOI2018] 染色

    BZOJ 5306 [HAOI2018] 染色 首先,求出$N$个位置,出现次数恰好为$S$的颜色至少有$K$种. 方案数显然为$a_i=\frac{n!\times (m-i)^{m-i\times ...

  2. 【BZOJ5306】 [Haoi2018]染色

    BZOJ5306 [Haoi2018]染色 Solution xzz的博客 代码实现 #include<stdio.h> #include<stdlib.h> #include ...

  3. [洛谷P4491] [HAOI2018]染色

    洛谷题目链接:[HAOI2018]染色 题目背景 HAOI2018 Round2 第二题 题目描述 为了报答小 C 的苹果, 小 G 打算送给热爱美术的小 C 一块画布, 这块画布可 以抽象为一个长度 ...

  4. [BZOJ5306] [HAOI2018]染色(容斥原理+NTT)

    [BZOJ5306] [HAOI2018]染色(容斥原理+NTT) 题面 一个长度为 n的序列, 每个位置都可以被染成 m种颜色中的某一种. 如果n个位置中恰好出现了 S次的颜色有 K种, 则小 C ...

  5. 【题解】[HAOI2018]染色(NTT+容斥/二项式反演)

    [题解][HAOI2018]染色(NTT+容斥/二项式反演) 可以直接写出式子: \[ f(x)={m \choose x}n!{(\dfrac 1 {(Sx)!})}^x(m-x)^{n-Sx}\d ...

  6. P4491 [HAOI2018]染色

    题目链接:洛谷 题目大意:$n$个位置染$m$种颜色,如果出现次数恰为$S$次的颜色有$k$种,则对答案有$W_k$的贡献,求所有染色方案的答案之和$\bmod 1004535809$. 数据范围:$ ...

  7. 【BZOJ5306】[HAOI2018]染色(NTT)

    [BZOJ5306]染色(NTT) 题面 BZOJ 洛谷 题解 我们只需要考虑每一个\(W[i]\)的贡献就好了 令\(lim=min(M,\frac{N}{S})\) 那么,开始考虑每一个\(W[i ...

  8. [BZOJ5306][HAOI2018]染色

    bzoj luogu Description 给一个长度为\(n\)的序列染色,每个位置上可以染\(m\)种颜色.如果染色后出现了\(S\)次的颜色有\(k\)种,那么这次染色就可以获得\(w_k\) ...

  9. P4491 [HAOI2018]染色 容斥+NTT

    $ \color{#0066ff}{ 题目描述 }$ 为了报答小 C 的苹果, 小 G 打算送给热爱美术的小 C 一块画布, 这块画布可 以抽象为一个长度为 \(N\) 的序列, 每个位置都可以被染成 ...

随机推荐

  1. BZOJ 1088 扫雷Mine 枚举初始状态

    题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=1088 题目大意: 现在棋盘是n×2的,第一列里面某些格子是雷,而第二列没有雷,如下图: ...

  2. programming-languages学习笔记--第4部分

    programming-languages学习笔记–第4部分 */--> pre.src {background-color: #292b2e; color: #b2b2b2;} program ...

  3. 解决MySQL新建用户后无法登录问题

    在PHPMyAdmin里创建了一个新的用户,并且创建了密码,但是却一直无法使用这个账户登录到MySQL里. 解决过程分享给大家~ 1.以root身份登录mysql 2.删除MySQL中默认存在一个用户 ...

  4. Kali-linux控制Meterpreter

    Meterpreter是Metasploit框架中的一个杀手锏,通常作为利用漏洞后的攻击载荷所使用,攻击载荷在触发漏洞后能够返回给用户一个控制通道.当使用Armitage.MSFCLI或MSFCONS ...

  5. 【Mybatis】参数处理

    单个参数:mybatis不会做特殊处理, #{参数名/任意名}:取出参数值. 多个参数:mybatis会做特殊处理. 多个参数会被封装成 一个map, key:param1...paramN,或者参数 ...

  6. Java SPI(Service Provider Interface)

    SPI 全称为 (Service Provider Interface) ,是JDK内置的一种服务提供发现机制. 目前有不少框架用它来做服务的扩展发现, 简单来说,它就是一种动态替换发现的机制, 举个 ...

  7. Dubbo实践(三)框架设计

    整体设计 图例说明: 图中左边淡蓝背景的为服务消费方使用的接口,右边淡绿色背景的为服务提供方使用的接口,位于中轴线上的为双方都用到的接口: 图中从下至上分为十层,各层均为单向依赖,右边的黑色箭头代表层 ...

  8. C#中Array类

    Array类是C#中所有数组的基类,它是在System命名空间中定义的,Array类提供了各种用于数组的属性和方法

  9. ssh框架错误:org.hibernate.LazyInitializationException: failed to lazily initialize a collection of role。

    在做ssh项目练习的时候出现问题: org.hibernate.LazyInitializationException: failed to lazily initialize a collectio ...

  10. SQL Server 数据库空间使用情况

    GO /****** Object: StoredProcedure [dbo].[SpaceUsed] Script Date: 2017-12-01 11:15:11 ******/ SET AN ...