#3456. 城市规划

设fnf_nfn​为nnn个点的的点的简单无向连通图数目,gng_ngn​为nnn个点的简单无向图个数(不要求联通)。

对于gng_ngn​显然有gn=2n(n−1)2g_n = 2 ^{\frac{n(n - 1)}{2}}gn​=22n(n−1)​,共有n(n+1)2\frac{n(n + 1)}{2}2n(n+1)​条边,然后每条边可选可不选。

我们枚举111所在的点的连通块可得:
gn=∑i=1nCn−1i−1fign−i2(2n)=∑i=1n(i−1n−1)fi2(2n−i)2(2n)=∑i=1n(n−1)!(i−1)!(n−i)!fi2(2n−i)2(2n)(n−1)!=∑i=1nfi(i−1)!2(2n−i)(n−i)!设G(x)=∑n=1∞2(2n)(n−1)!xnF(x)=∑n=1∞fn(n−1)!H(x)=∑n=0∞2(2n)n!G(x)=F(x)H(x)F(x)=G(x)H−1(x)构造多项式,多项式求逆,把[xn]项系数乘上(n−1)!即是答案g_n = \sum\limits_{i = 1} ^{n}C_{n - 1} ^{i - 1}f_ig_{n - i}\\ 2 ^{(_2 ^ n)} = \sum_{i = 1} ^{n}(_{i - 1} ^{n - 1})f_i 2 ^{(_2 ^{n - i})}\\ 2 ^{(_2 ^ n)} = \sum_{i = 1} ^{n} \frac{(n - 1)!}{(i - 1)!(n - i)!} f_i 2 ^{(_2 ^{n - i})}\\ \frac{2 ^{(_2 ^ n)}}{(n - 1)!} = \sum_{i = 1} ^{n} \frac{f_i}{(i - 1)!} \frac{2^{(_2 ^{n - i})}}{(n - i)!}\\ 设G(x) = \sum_{n = 1} ^{\infty} \frac{2 ^{(_2 ^n)}}{(n - 1)!} x ^ n\\ F(x) = \sum_{n = 1} ^{\infty} \frac{f_n}{(n - 1)!}\\ H(x) = \sum_{n = 0} ^{\infty} \frac{2 ^{(_2 ^n)}}{n!}\\ G(x) = F(x) H(x)\\ F(x) = G(x)H^{-1}(x)\\ 构造多项式,多项式求逆,把[x ^ n]项系数乘上(n - 1)!即是答案\\ gn​=i=1∑n​Cn−1i−1​fi​gn−i​2(2n​)=i=1∑n​(i−1n−1​)fi​2(2n−i​)2(2n​)=i=1∑n​(i−1)!(n−i)!(n−1)!​fi​2(2n−i​)(n−1)!2(2n​)​=i=1∑n​(i−1)!fi​​(n−i)!2(2n−i​)​设G(x)=n=1∑∞​(n−1)!2(2n​)​xnF(x)=n=1∑∞​(n−1)!fn​​H(x)=n=0∑∞​n!2(2n​)​G(x)=F(x)H(x)F(x)=G(x)H−1(x)构造多项式,多项式求逆,把[xn]项系数乘上(n−1)!即是答案

#include <bits/stdc++.h>using namespace std;const int mod = 1004535809, inv2 = mod + 1 >> 1;namespace Quadratic_residue {struct Complex {int r, i;Complex(int _r = 0, int _i = 0) : r(_r), i(_i) {}};int I2;Complex operator * (const Complex &a, Complex &b) {return Complex((1ll * a.r * b.r % mod  + 1ll * a.i * b.i % mod * I2 % mod) % mod, (1ll * a.r * b.i % mod + 1ll * a.i * b.r % mod) % mod);}Complex quick_pow(Complex a, int n) {Complex ans = Complex(1, 0);while (n) {if (n & 1) {ans = ans * a;}a = a * a;n >>= 1;}return ans;}int get_residue(int n) {mt19937 e(233);if (n == 0) {return 0;}if(quick_pow(n, (mod - 1) >> 1).r == mod - 1) {return -1;}uniform_int_distribution<int> r(0, mod - 1);int a = r(e);while(quick_pow((1ll * a * a % mod - n + mod) % mod, (mod - 1) >> 1).r == 1) {a = r(e);}I2 = (1ll * a * a % mod - n + mod) % mod;int x = quick_pow(Complex(a, 1), (mod + 1) >> 1).r, y = mod - x;if(x > y) swap(x, y);return x;}
}const int N = 1e6 + 10;int r[N], inv[N], b[N], c[N], d[N], e[N], t[N];int quick_pow(int a, int n) {int ans = 1;while (n) {if (n & 1) {ans = 1ll * a * ans % mod;}a = 1ll * a * a % mod;n >>= 1;}return ans;
}void get_r(int lim) {for (int i = 0; i < lim; i++) {r[i] = (i & 1) * (lim >> 1) + (r[i >> 1] >> 1);}
}void get_inv(int n) {inv[1] = 1;for (int i = 2; i <= n; i++) {inv[i] = 1ll * (mod - mod / i) * inv[mod % i] % mod;}
}void NTT(int *f, int lim, int rev) {for (int i = 0; i < lim; i++) {if (i < r[i]) {swap(f[i], f[r[i]]);}}for (int mid = 1; mid < lim; mid <<= 1) {int wn = quick_pow(3, (mod - 1) / (mid << 1));for (int len = mid << 1, cur = 0; cur < lim; cur += len) {int w = 1;for (int k = 0; k < mid; k++, w = 1ll * w * wn % mod) {int x = f[cur + k], y = 1ll * w * f[cur + mid + k] % mod;f[cur + k] = (x + y) % mod, f[cur + mid + k] = (x - y + mod) % mod;}}}if (rev == -1) {int inv = quick_pow(lim, mod - 2);reverse(f + 1, f + lim);for (int i = 0; i < lim; i++) {f[i] = 1ll * f[i] * inv % mod;}}
}void polyinv(int *f, int *g, int n) {if (n == 1) {g[0] = quick_pow(f[0], mod - 2);return ;}polyinv(f, g, n + 1 >> 1);for (int i = 0; i < n; i++) {t[i] = f[i];}int lim = 1;while (lim < 2 * n) {lim <<= 1;}get_r(lim);NTT(t, lim, 1);NTT(g, lim, 1);for (int i = 0; i < lim; i++) {int cur = (2 - 1ll * g[i] * t[i] % mod + mod) % mod;g[i] = 1ll * g[i] * cur % mod;t[i] = 0;}NTT(g, lim, -1);for (int i = n; i < lim; i++) {g[i] = 0;}
}void polysqrt(int *f, int *g, int n) {if (n == 1) {g[0] = Quadratic_residue::get_residue(f[0]);return ;}polysqrt(f, g, n + 1 >> 1);polyinv(g, b, n);int lim = 1;while (lim < 2 * n) {lim <<= 1;}get_r(lim);for (int i = 0; i < n; i++) {t[i] = f[i];}NTT(g, lim, 1);NTT(b, lim, 1);NTT(t, lim, 1);for (int i = 0; i < lim; i++) {g[i] = (1ll * inv2 * g[i] % mod + 1ll * inv2 * b[i] % mod * t[i] % mod) % mod;b[i] = t[i] = 0;}NTT(g, lim, -1);for (int i = n; i < lim; i++) {g[i] = 0;}
}void derivative(int *a, int *b, int n) {for (int i = 0; i < n; i++) {b[i] = 1ll * a[i + 1] * (i + 1) % mod;}
}void integrate(int *a, int n) {for (int i = n - 1; i >= 1; i--) {a[i] = 1ll * a[i - 1] * inv[i] % mod;}a[0] = 0;
}void polyln(int *f, int *g, int n) {polyinv(f, b, n);derivative(f, g, n);int lim = 1;while (lim < 2 * n) {lim <<= 1;}get_r(lim);NTT(g, lim, 1);NTT(b, lim, 1);for (int i = 0; i < lim; i++) {g[i] = 1ll * g[i] * b[i] % mod;b[i] = 0;}NTT(g, lim, -1);for (int i = n; i < lim; i++) {g[i] = 0;}integrate(g, n);
}void polyexp(int *f, int *g, int n) {if (n == 1) {g[0] = 1;return ;}polyexp(f, g, n + 1 >> 1);int lim = 1;while (lim < 2 * n) {lim <<= 1;}polyln(g, d, n);for (int i = 0; i < n; i++) {t[i] = (f[i] - d[i] + mod) % mod;}t[0] = (t[0] + 1) % mod;get_r(lim);NTT(g, lim, 1);NTT(t, lim, 1);for (int i = 0; i < lim; i++) {g[i] = 1ll * g[i] * t[i] % mod;t[i] = d[i] =  0;}NTT(g, lim, -1);for (int i = n; i < lim; i++) {g[i] = 0;}
}/*b存放多项式逆,c存放多项式开根,d存放多项式对数ln,e存放多项式指数exp,t作为中间转移数组,如果要用到polyln,得提前调用get_inv(n)先预先得到我们想要得到的逆元范围。
*/int h[N], g[N], fac[N], ifac[N], n;void init() {fac[0] = 1;for (int i = 1; i < N; i++) {fac[i] = 1ll * i * fac[i - 1] % mod;}ifac[N - 1] = quick_pow(fac[N - 1], mod - 2);for (int i = N - 2; i >= 0; i--) {ifac[i] = 1ll * ifac[i + 1] * (i + 1) % mod;}
}int main() {// freopen("in.txt", "r", stdin);// freopen("out.txt", "w", stdout);// ios::sync_with_stdio(false), cin.tie(0), cout.tie(0);scanf("%d", &n);init();for (int i = 1; i <= n; i++) {g[i] = 1ll * quick_pow(2, 1ll * i * (i - 1) / 2 % (mod - 1)) * ifac[i - 1] % mod;}for (int i = 0; i <= n; i++) {h[i] = 1ll * quick_pow(2, 1ll * i * (i - 1) / 2 % (mod - 1)) * ifac[i] % mod;}polyinv(h, b, n + 1);for (int i = 0; i <= n; i++) {h[i] = b[i];b[i] = 0;}int lim = 1;while (lim <= 2 * n) {lim <<= 1;}get_r(lim);NTT(g, lim, 1);NTT(h, lim, 1);for (int i = 0; i < lim; i++) {g[i] = 1ll * g[i] * h[i] % mod;h[i] = 0;}NTT(g, lim, -1);printf("%d\n", 1ll * g[n] * fac[n - 1] % mod);return 0;
}

#3456. 城市规划(生成函数,多项式求逆)相关推荐

  1. 【BZOJ】3456: 城市规划 动态规划+多项式求逆

    [题意]求n个点的带标号无向连通图个数 mod 1004535809.n<=130000. [算法]动态规划+多项式求逆 [题解]设$g_n$表示n个点的无向图个数,那么显然 $$g_n=2^{ ...

  2. BZOJ 3456: 城市规划(dp+多项式求逆)

    传送门 解题思路 这道题就是求带标号的无向连通图个数,首先考虑\(O(n^2)\)的做法,设\(f_i\)表示有\(i\)个节点的无向连通图个数,那么考虑容斥,先把所有的无向图求出,即为\(2^{C( ...

  3. (每日一题)P4841 [集训队作业2013]城市规划 (无向连通图计数)(普通生成函数 + 多项式求逆)

    整理的算法模板合集: ACM模板 点我看算法全家桶系列!!! 实际上是一个全新的精炼模板整合计划 每日一题(莫反 / 多项式 / 母函数 / 群论) 2021.4.14 生成函数 + 多项式求逆 Pr ...

  4. CF438E The Child and Binary Tree(有意思的生成函数 + 多项式求逆 + 多项式开方)

    整理的算法模板合集: ACM模板 点我看多项式全家桶(●^◡_◡◡​^●) CF438E The Child and Binary Tree 简单的黑题 首先我们发现模数为99824435399824 ...

  5. 【BZOJ】3456: 城市规划(多项式求ln)

    题解 在我写过分治NTT,多项式求逆之后 我又一次写了多项式求ln 我们定义一个数列的指数型生成函数为 \(\sum_{i = 0}^{n} \frac{A_{i}}{i!} x^{i}\) 然后这个 ...

  6. 「Luogu4233」射命丸文的笔记-生成函数+多项式求逆

    Description 链接 Solution 考虑所有竞赛图的哈密顿回路条数n!n2Cn2−n\frac {n!} {n} 2^{C_{n}^{2}-n}nn!​2Cn2​−n,即选出一条哈密顿回路 ...

  7. HDU 5730 Shell Necklace(生成函数 多项式求逆)

    Shell Necklace 由题意可得f[n]=∑i=1na[i]f[n−i]f[n] = \sum\limits_{i = 1} ^{n} a[i] f[n - i]f[n]=i=1∑n​a[i] ...

  8. 洛谷P4841 城市规划(多项式求逆)

    传送门 这题太珂怕了--如果是我的话完全想不出来-- 题解 1 //minamoto 2 #include<iostream> 3 #include<cstdio> 4 #in ...

  9. P6295-有标号 DAG 计数【多项式求逆,多项式ln】

    正题 题目链接:https://www.luogu.com.cn/problem/P6295 题目大意 求所有nnn个点的弱联通DAGDAGDAG数量. 1≤n≤1051\leq n\leq 10^5 ...

最新文章

  1. 分享一个mysql 复杂查询的例子
  2. Cordova+jQuery Mobile+Spring REST
  3. 3_深度学习中显卡的使用和显存的分配(20181213)
  4. ConfigUtil读取配置文件
  5. python 学习中遇到的问题(持续更新中)
  6. jQuery hash 插件
  7. pos机改造迷你打印机_小票打印机如何自动弹出钱箱
  8. wxpython多线程 假死_wxpython多线程防假死与线程间传递消息实例详解
  9. kvm+libvirt虚拟机快照浅析[转]
  10. Linux之执行一个可执行文件
  11. 重磅:GB/T 35273-2020《信息安全技术个人信息安全规范》最新解读
  12. 台达内部速度指令_台达PLC连续脉冲输出如何控制速度
  13. 新浪php工程师面试题
  14. python中seek函数_Python seek()函数
  15. 【计算机二级Python】模拟试卷第4套选择题
  16. 24、将Div中的所有元素保存为图片 Html2Canvas
  17. ios java 程序_使用java代码实现推送IOS消息
  18. 树突状细胞(DC细胞)特征及应用进展综述
  19. JAVA 使用POI读取文档
  20. 澳洲电源和电池充电器对应标准的公告将强制执行2022年6月15日起

热门文章

  1. jq取第一个子元素为select_【转】jquery如何获取第一个或最后一个子元素?
  2. 可以自发热的袜子,穿上暖3.9℃,这个冬天不再怕脚冷!
  3. 为什么每个理发店门口都有彩色的柱子?你不知道吧
  4. 人为什么会出轨?麻省理工学院告诉你:男女配对的真相
  5. oracle分区exchange,oracle 分区表exchange原理
  6. python怎么获取lol皮肤名称_LOL手游免费皮肤获得方法 LOL手游皮肤怎么获得
  7. bpmn如何查看代码 idea_提高程序员效率的IDEA插件推荐(五大神器)
  8. TDengine和DolphinDB哪个更好,哈哈哈哈,闲来无聊分析了一下。
  9. python退出帮助系统help应该使用exit_python--help - tesion
  10. android 监听布局改变,Android通过监听最外层布局的改变监听键盘的状态,软键盘的弹出和收起都会改变外层布局(前提是把Activity的mode设置成压缩);...