\(\mathcal{Description}\)

  Link.

  用 \(m\) 种颜色为长为 \(n\) 的序列染色,每个位置一种颜色。对于一种染色方案,其价值为 \(w(\text{出现恰 }s\text{ 次的颜色种数})\)(\(w(0..m)\) 给定),求所有染色方案的价值和。

  \(n\le10^7\),\(m\le10^5\),答案对 \(p=1004535809=479\times2^{21}+1\) 取模。

\(\mathcal{Solution}\)

  记 \(l=\min\{m,\lfloor\frac{n}{s}\rfloor\}\),显然只能对于 \(i=0..l\),分别算出 \(w(i)\) 的贡献次数。考虑容斥,令 \(f(i)\) 表示至少 \(i\) 种颜色恰好出现 \(s\) 次的染色方案数。那么:

\[f(i)=\binom{m}i\frac{n!}{(s!)^i(n-is)!}(m-i)^{n-is}
\]

  即:选出 \(i\) 种颜色钦定恰好出现 \(s\) 次;多重集排列安排已被染色的 \(is\) 个位置和剩下的 \((n-is)\) 个位置;为 \((n-is)\) 个位置任选未使用的颜色。

  此后,令 \(g(i)\) 表示恰好 \(i\) 种颜色恰好出现 \(s\) 次的染色方案数,容斥:

\[\begin{aligned}
g(i)&=\sum_{j=i}^l(-1)^{l-i}\binom{j}if_j\\
&=\sum_{j=i}^l(-1)^{l-j}\frac{j!}{i!(j-i)!}f_j\\
&=\frac{1}{i!}\sum_{j=i}^l\frac{(-1)^{l-j}}{(j-i)!}(j!f_j)
\end{aligned}
\]

  很俗套 trick,翻转乘积式任意一项就能写成卷积形式,NTT 求出 \(g(i)\) 即可。

  复杂度 \(\mathcal O(n+l\log l)\)。

\(\mathcal{Code}\)

  1. /* Clearink */
  2. #include <cstdio>
  3. #define rep( i, l, r ) for ( int i = l, rpbound##i = r; i <= rpbound##i; ++i )
  4. #define per( i, r, l ) for ( int i = r, rpbound##i = l; i >= rpbound##i; --i )
  5. const int MAXN = 1e7, MAXM = 1e5, MOD = 1004535809, G = 3, MAXLEN = 1 << 18;
  6. int n, m, s, w[MAXM + 5], f[MAXLEN + 5], g[MAXLEN + 5], rev[MAXLEN + 5];
  7. int fac[MAXN + 5], ifac[MAXN + 5]; // warning: 77MB.
  8. inline char fgc () {
  9. static char buf[1 << 17], *p = buf, *q = buf;
  10. return p == q && ( q = buf + fread ( p = buf, 1, 1 << 17, stdin ), p == q )
  11. ? EOF : *p++;
  12. }
  13. inline int rint () {
  14. int x = 0; char s = fgc ();
  15. for ( ; s < '0' || '9' < s; s = fgc () );
  16. for ( ; '0' <= s && s <= '9'; s = fgc () ) x = x * 10 + ( s ^ '0' );
  17. return x;
  18. }
  19. inline void iswp ( int& a, int& b ) { a ^= b ^= a ^= b; }
  20. inline int imin ( const int a, const int b ) { return a < b ? a : b; }
  21. inline int imax ( const int a, const int b ) { return a < b ? b : a; }
  22. inline int mul ( const long long a, const int b ) { return a * b % MOD; }
  23. inline int sub ( int a, const int b ) { return ( a -= b ) < 0 ? a + MOD : a; }
  24. inline int add ( int a, const int b ) { return ( a += b ) < MOD ? a : a - MOD; }
  25. inline int mpow ( int a, int b ) {
  26. int ret = 1;
  27. for ( ; b; a = mul ( a, a ), b >>= 1 ) ret = mul ( ret, b & 1 ? a : 1 );
  28. return ret;
  29. }
  30. inline int inv ( const int n ) { return mpow ( n, MOD - 2 ); }
  31. inline void init ( const int n ) {
  32. fac[0] = 1;
  33. rep ( i, 1, n ) fac[i] = mul ( i, fac[i - 1] );
  34. ifac[n] = inv ( fac[n] );
  35. per ( i, n - 1, 0 ) ifac[i] = mul ( i + 1, ifac[i + 1] );
  36. }
  37. inline int comb ( const int n, const int m ) {
  38. return n < m ? 0 : mul ( fac[n], mul ( ifac[m], ifac[n - m] ) );
  39. }
  40. inline void NTT ( const int n, int* A, const int flg ) {
  41. rep ( i, 0, n - 1 ) if ( i < rev[i] ) iswp ( A[i], A[rev[i]] );
  42. for ( int i = 2, stp = 1; i <= n; i <<= 1, stp <<= 1 ) {
  43. int w = mpow ( G, ( MOD - 1 ) / i );
  44. if ( !~flg ) w = inv ( w );
  45. for ( int j = 0; j < n; j += i ) {
  46. for ( int k = j, wk = 1; k < j + stp; ++k, wk = mul ( wk, w ) ) {
  47. int ev = A[k], ov = mul ( wk, A[k + stp] );
  48. A[k] = add ( ev, ov ), A[k + stp] = sub ( ev, ov );
  49. }
  50. }
  51. }
  52. if ( !~flg ) {
  53. int in = inv ( n );
  54. rep ( i, 0, n - 1 ) A[i] = mul ( in, A[i] );
  55. }
  56. }
  57. int main () {
  58. n = rint (), m = rint (), s = rint ();
  59. init ( imax ( n, m ) );
  60. rep ( i, 0, m ) w[i] = rint ();
  61. int mx = imin ( m, n / s );
  62. for ( int i = 0, fp = 1; i <= mx; ++i, fp = mul ( fp, fac[s] ) ) {
  63. f[i] = mul ( fac[i], mul ( mul ( comb ( m, i ), mpow ( m - i, n - i * s ) ),
  64. mul ( fac[n], inv ( mul ( fp, fac[n - i * s] ) ) ) ) );
  65. }
  66. rep ( i, 0, mx ) {
  67. g[i] = ( ( mx - i ) & 1 ? sub : add )( 0, inv ( fac[mx - i] ) );
  68. }
  69. int len = 1, lgv = 0;
  70. for ( ; len < mx + 1 << 1; len <<= 1, ++lgv );
  71. rep ( i, 0, len - 1 ) rev[i] = ( rev[i >> 1] >> 1 ) | ( ( i & 1 ) << lgv >> 1 );
  72. NTT ( len, f, 1 ), NTT ( len, g, 1 );
  73. rep ( i, 0, len - 1 ) f[i] = mul ( f[i], g[i] );
  74. NTT ( len, f, -1 );
  75. int ans = 0;
  76. rep ( i, 0, mx ) {
  77. ans = add ( ans, mul ( w[i], mul ( f[i + mx], ifac[i] ) ) );
  78. }
  79. printf ( "%d\n", ans );
  80. return 0;
  81. }

Solution -「HAOI 2018」「洛谷 P4491」染色的更多相关文章

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

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

  2. 「区间DP」「洛谷P1043」数字游戏

    「洛谷P1043」数字游戏 日后再写 代码 /*#!/bin/sh dir=$GEDIT_CURRENT_DOCUMENT_DIR name=$GEDIT_CURRENT_DOCUMENT_NAME ...

  3. Solution -「JSOI 2019」「洛谷 P5334」节日庆典

    \(\mathscr{Description}\)   Link.   给定字符串 \(S\),求 \(S\) 的每个前缀的最小表示法起始下标(若有多个,取最小的).   \(|S|\le3\time ...

  4. Solution -「洛谷 P4372」Out of Sorts P

    \(\mathcal{Description}\)   OurOJ & 洛谷 P4372(几乎一致)   设计一个排序算法,设现在对 \(\{a_n\}\) 中 \([l,r]\) 内的元素排 ...

  5. Solution -「POI 2010」「洛谷 P3511」MOS-Bridges

    \(\mathcal{Description}\)   Link.(洛谷上这翻译真的一言难尽呐.   给定一个 \(n\) 个点 \(m\) 条边的无向图,一条边 \((u,v,a,b)\) 表示从 ...

  6. Solution -「APIO 2016」「洛谷 P3643」划艇

    \(\mathcal{Description}\)   Link & 双倍经验.   给定 \(n\) 个区间 \([a_i,b_i)\)(注意原题是闭区间,这里只为方便后文描述),求 \(\ ...

  7. 「洛谷4197」「BZOJ3545」peak【线段树合并】

    题目链接 [洛谷] [BZOJ]没有权限号嘤嘤嘤.题号:3545 题解 窝不会克鲁斯卡尔重构树怎么办??? 可以离线乱搞. 我们将所有的操作全都存下来. 为了解决小于等于\(x\)的操作,那么我们按照 ...

  8. 「洛谷3338」「ZJOI2014」力【FFT】

    题目链接 [BZOJ] [洛谷] 题解 首先我们需要对这个式子进行化简,否则对着这么大一坨东西只能暴力... \[F_i=\sum_{j<i} \frac{q_iq_j}{(i-j)^2}-\s ...

  9. 「BZOJ2733」「洛谷3224」「HNOI2012」永无乡【线段树合并】

    题目链接 [洛谷] 题解 很明显是要用线段树合并的. 对于当前的每一个连通块都建立一个权值线段树. 权值线段树处理操作中的\(k\)大的问题. 如果需要合并,那么就线段树暴力合并,时间复杂度是\(nl ...

随机推荐

  1. redis 加锁与解锁的详细总结,解决线程并发导致脏数据

    1.前言 对每个controller来说都是全新且单独的,原因是多线程,如果多个请求操作共有的数据,这样的并发操作会导致脏数据 怎么解决? mysql可以使用积极锁解决, 这里讲解的是redis的解决 ...

  2. 使用.NET 6开发TodoList应用(25)——实现RefreshToken

    系列导航及源代码 使用.NET 6开发TodoList应用文章索引 需求 在上一篇文章使用.NET 6开发TodoList应用(24)--实现基于JWT的Identity功能中,我们演示了如何使用.N ...

  3. 一文看懂B端产品和C端产品

    大纲 什么是B端产品 什么是C端产品 为什么会产生B端产品和C端产品 怎么判断一个产品是B端还是C端 B端产品和C端产品存在哪些差异 C端产品经理如何向B端产品经理转型 写在最后   什么是B, Bu ...

  4. Whitelabel Error Page错误原因

    前言: 今天在做项目中遇到了一个问题,项目启动成功,但是前段访问接口始终访问不成功,页面一直在404,百度了一番无非两种解决方案: 一.解决方案 1.项目是boot项目查看启动类的位置是否放置正确 要 ...

  5. 阿里神器 Seata 实现 TCC模式 解决分布式事务,真香!

    今天这篇文章介绍一下Seata如何实现TCC事务模式,文章目录如下: 什么是TCC模式? TCC(Try Confirm Cancel)方案是一种应用层面侵入业务的两阶段提交.是目前最火的一种柔性事务 ...

  6. 理解Cookie和Session机制,及其安全问题

    大家常说"Cookie保存在客户端而Session保存在服务端",很多人看了有疑惑,明明Session就在Cookie中啊,为什么这么说?二者到底有啥区别? 一.Cookie 首先 ...

  7. java之类的抽取与对象的创建

    Java语言之类的抽取 前言:世界由什么组成?This is a question.有人说是原子.分子,有人说是山川草木. 诚然,一千个人眼中有一千个哈姆雷特.而在程序员眼中,万物皆对象. 定义: 在 ...

  8. 【刷题-LeetCode】209. Minimum Size Subarray Sum

    Minimum Size Subarray Sum Given an array of n positive integers and a positive integer s, find the m ...

  9. Ajax的IE缓存问题

    Ajax之IE缓存问题 <!-- IE浏览器会对ajax的结果进行一个缓存,这样就会导致一个缓存问题 浏览器会读取缓存 而不会去使用一个新的数据 这样对一个时效性比较强的场景 ajax的缓存会影 ...

  10. golang中结构体当做函数参数或函数返回值都会被拷贝

    1. 结构体做函数的参数或返回值时,都会被重新拷贝一份如果不想拷贝,可以传递结构体指针 package main import "fmt" type Person struct { ...