\(\mathcal{Description}\)

  link.

  求有 \(n\) 个结点的点双连通图的个数,对 \(998244353\) 取模。

  \(n\le10^5\)。

\(\mathcal{Solution}\)

  奇怪的 GF 增加了 w!

  对于带标号简单无向图,其 \(\text{EGF}\) 为 \(F(x)=\displaystyle\sum_{i=0}^{+\infty}\frac{2^{i\choose2}x^i}{i!}\)(任意两点间有连与不连两种情况。)在此基础上,我们要求图连通(注意这里不是点双连通),即对于带标号简单连通无向图,\(\text{EGF}\) 有 \(G(x)=\ln F(x)\)。

  以下规定所有的图都是带标号的简单无向图。设有根连通图的 \(\text{EGF}\) 为 \(D(x)\),显然 \(D(x)=nG(x)\)。并设 \(i\) 个结点的点双连通图个数为 \(b_i\)。考虑任意一个简单无向图的根,它可能被包含在多个点双中。首先特判掉 \(n=1\) ——单点的情况。现在对于每一个不是根且在点双连通分量上的点,我们都可以在上面插上一个以其为根的无向连通图,并且不会影响到包含原来的根的任何点双连通分量的大小。所以每一个点双块的 \(\text{EGF}\) 是:

\[\sum_{i=1}\frac{b_{i+1}D^i(x)}{i!}
\]

  令 \(B(x)=\sum_{i=0}^{+\infty}b_{i+1}\frac{x^i}{i!}\),我们反过来表示 \(D(x)\),则:

\[D(x)=x\exp B\left(D(x)\right)
\]

  开始推式子,先对上式变形:

\[B\left(D(x)\right)=\ln\frac{D(x)}x
\]

  令 \(D^{-1}\) 是 \(D\) 的复合逆,代入得:

\[B(x)=\ln\frac{x}{D^{-1}(x)}
\]

  令 \(H(x)=\ln\frac{D(x)}x\),那么有 \(B(x)=H\left(D^{-1}(x)\right)\)。利用扩展拉格朗日反演,有:

\[[x^n]B(x)=\frac{1}n[x^{n-1}]H'(x)\left(\frac{x}{D(x)}\right)^n
\]

  把后面的多项式幂变形,并交换分子分母以便运算,得:

\[[x^n]B(x)=\frac{1}n[x^{n-1}]H'(x)\exp\left(-n\ln\frac{D(x)}x\right)
\]

  \(D\) 易求,那么整个式子都能用亿堆多项式模板算出来。复杂度 \(\mathcal O(n\log n)\)。

\(\mathcal{Code}\)

#include <cmath>
#include <cstdio> const int MAXN = 1 << 18, MOD = 998244353;
int n, fac[MAXN + 5], ifac[MAXN + 5], inv[MAXN + 5], F[MAXN + 5], G[MAXN + 5];
int H[MAXN + 5], lH[MAXN + 5], dH[MAXN + 5]; inline int qkpow ( int a, int b, const int p = MOD ) {
int ret = 1;
for ( ; b; a = 1ll * a * a % p, b >>= 1 ) ret = 1ll * ret * ( b & 1 ? a : 1 ) % p;
return ret;
} namespace Poly { const int G = 3; inline void NTT ( const int n, int* A, const int tp ) {
static int lstn = -1, rev[MAXN + 5] {};
if ( lstn ^ n ) {
int lgn = log ( n ) / log ( 2 ) + 0.5;
for ( int i = 0; i < n; ++ i ) rev[i] = ( rev[i >> 1] >> 1 ) | ( ( i & 1 ) << lgn >> 1 );
lstn = n;
}
for ( int i = 0; i < n; ++ i ) if ( i < rev[i] ) A[i] ^= A[rev[i]] ^= A[i] ^= A[rev[i]];
for ( int i = 2, stp = 1; i <= n; i <<= 1, stp <<= 1 ) {
int w = qkpow ( G, ( MOD - 1 ) / i );
if ( ! ~ tp ) w = qkpow ( w, MOD - 2 );
for ( int j = 0; j < n; j += i ) {
for ( int k = j, r = 1; k < j + stp; ++ k, r = 1ll * r * w % MOD ) {
int ev = A[k], ov = 1ll * r * A[k + stp] % MOD;
A[k] = ( ev + ov ) % MOD, A[k + stp] = ( ev - ov + MOD ) % MOD;
}
}
}
if ( ! ~ tp ) for ( int i = 0; i < n; ++ i ) A[i] = 1ll * A[i] * inv[n] % MOD;
} inline void polyDir ( const int n, const int* A, int* R ) {
for ( int i = 1; i < n; ++ i ) R[i - 1] = 1ll * i * A[i] % MOD;
R[n - 1] = 0;
} inline void polyInt ( const int n, const int* A, int* R ) {
for ( int i = n - 1; ~ i; -- i ) R[i + 1] = 1ll * inv[i + 1] * A[i] % MOD;
R[0] = 0;
} inline void polyInv ( const int n, const int* A, int* R ) {
static int tmp[2][MAXN + 5] {};
if ( n == 1 ) return void ( R[0] = qkpow ( A[0], MOD - 2 ) );
polyInv ( n >> 1, A, R );
for ( int i = 0; i < n; ++ i ) tmp[0][i] = A[i], tmp[1][i] = R[i];
NTT ( n << 1, tmp[0], 1 ), NTT ( n << 1, tmp[1], 1 );
for ( int i = 0; i < n << 1; ++ i ) tmp[0][i] = 1ll * tmp[0][i] * tmp[1][i] % MOD * tmp[1][i] % MOD;
NTT ( n << 1, tmp[0], -1 );
for ( int i = 0; i < n; ++ i ) R[i] = ( 2ll * R[i] % MOD - tmp[0][i] + MOD ) % MOD;
for ( int i = 0; i < n << 1; ++ i ) tmp[0][i] = tmp[1][i] = 0;
} inline void polyLn ( const int n, const int* A, int* R ) {
static int tmp[2][MAXN + 5] {};
polyDir ( n, A, tmp[0] ), polyInv ( n, A, tmp[1] );
NTT ( n << 1, tmp[0], 1 ), NTT ( n << 1, tmp[1], 1 );
for ( int i = 0; i < n << 1; ++ i ) tmp[0][i] = 1ll * tmp[0][i] * tmp[1][i] % MOD;
NTT ( n << 1, tmp[0], -1 ), polyInt ( n << 1, tmp[0], R );
for ( int i = 0; i < n << 1; ++ i ) tmp[0][i] = tmp[1][i] = 0;
} inline void polyExp ( const int n, const int* A, int* R ) {
static int tmp[MAXN + 5] {};
if ( n == 1 ) return void ( R[0] = 1 );
polyExp ( n >> 1, A, R ), polyLn ( n, R, tmp );
tmp[0] = ( A[0] + 1 - tmp[0] + MOD ) % MOD;
for ( int i = 1; i < n; ++ i ) tmp[i] = ( A[i] - tmp[i] + MOD ) % MOD;
NTT ( n << 1, tmp, 1 ), NTT ( n << 1, R, 1 );
for ( int i = 0; i < n << 1; ++ i ) R[i] = 1ll * R[i] * tmp[i] % MOD;
NTT ( n << 1, R, -1 );
for ( int i = n; i < n << 1; ++ i ) R[i] = tmp[i] = 0;
} } // namespace Poly. inline void init () {
inv[1] = fac[0] = ifac[0] = fac[1] = ifac[1] = 1;
for ( int i = 2; i <= MAXN; ++ i ) {
fac[i] = 1ll * i * fac[i - 1] % MOD;
inv[i] = 1ll * ( MOD - MOD / i ) * inv[MOD % i] % MOD;
ifac[i] = 1ll * inv[i] * ifac[i - 1] % MOD;
}
int len = MAXN >> 1;
for ( int i = 0; i < len; ++ i ) F[i] = 1ll * qkpow ( 2, ( i * ( i - 1ll ) >> 1 ) % ( MOD - 1 ) ) * ifac[i] % MOD;
Poly::polyLn ( len, F, G );
for ( int i = 0; i < len; ++ i ) G[i] = 1ll * i * G[i] % MOD;
for ( int i = 0; i < len - 1; ++ i ) G[i] = G[i + 1];
G[len - 1] = 0;
Poly::polyLn ( len, G, H ), Poly::polyDir ( len, H, dH );
Poly::NTT ( MAXN, dH, 1 );
} inline void solve () {
int len = MAXN >> 1;
if ( ! -- n ) return void ( puts ( "1" ) );
for ( int i = 0; i < MAXN; ++ i ) F[i] = G[i] = 0;
for ( int i = 0; i < len; ++ i ) F[i] = 1ll * ( MOD - n ) % MOD * H[i] % MOD;
Poly::polyExp ( len, F, G ), Poly::NTT ( MAXN, G, 1 );
for ( int i = 0; i < MAXN; ++ i ) G[i] = 1ll * dH[i] * G[i] % MOD;
Poly::NTT ( MAXN, G, -1 );
printf ( "%d\n", int ( 1ll * inv[n] * fac[n] % MOD * G[n - 1] % MOD ) );
} int main () {
init ();
for ( int i = 1; i <= 5; ++ i ) scanf ( "%d", &n ), solve ();
return 0;
}

Solution -「洛谷 P5827」点双连通图计数的更多相关文章

  1. Solution -「洛谷 P5827」边双连通图计数

    \(\mathcal{Description}\)   link.   求包含 \(n\) 个点的边双连通图的个数.   \(n\le10^5\). \(\mathcal{Solution}\)    ...

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

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

  3. Note/Solution -「洛谷 P5158」「模板」多项式快速插值

    \(\mathcal{Description}\)   Link.   给定 \(n\) 个点 \((x_i,y_i)\),求一个不超过 \(n-1\) 次的多项式 \(f(x)\),使得 \(f(x ...

  4. Solution -「洛谷 P5236」「模板」静态仙人掌

    \(\mathcal{Description}\)   Link.   给定一个 \(n\) 个点 \(m\) 条边的仙人掌,\(q\) 组询问两点最短路.   \(n,q\le10^4\),\(m\ ...

  5. Solution -「洛谷 P4198」楼房重建

    \(\mathcal{Description}\)   Link.   给定点集 \(\{P_n\}\),\(P_i=(i,h_i)\),\(m\) 次修改,每次修改某个 \(h_i\),在每次修改后 ...

  6. Solution -「洛谷 P6577」「模板」二分图最大权完美匹配

    \(\mathcal{Description}\)   Link.   给定二分图 \(G=(V=X\cup Y,E)\),\(|X|=|Y|=n\),边 \((u,v)\in E\) 有权 \(w( ...

  7. Solution -「洛谷 P6021」洪水

    \(\mathcal{Description}\)   Link.   给定一棵 \(n\) 个点的带点权树,删除 \(u\) 点的代价是该点点权 \(a_u\).\(m\) 次操作: 修改单点点权. ...

  8. Solution -「洛谷 P4719」「模板」"动态 DP" & 动态树分治

    \(\mathcal{Description}\)   Link.   给定一棵 \(n\) 个结点的带权树,\(m\) 次单点点权修改,求出每次修改后的带权最大独立集.   \(n,m\le10^5 ...

  9. Solution -「洛谷 P4320」道路相遇

    \(\mathcal{Description}\)   Link.   给定一个 \(n\) 个点 \(m\) 条边的连通无向图,并给出 \(q\) 个点对 \((u,v)\),询问 \(u\) 到 ...

随机推荐

  1. [ unittest ] 使用初体验

    import unittest from cal import Calculate class Mytest(unittest.TestCase): def setUp(self): self.cal ...

  2. websocket 使用 spring 的service层 ,进而调用里面的 dao层 来操作数据库 ,包括redis、mysql等通用

    1.前言 描述一下今天用websocket踩得坑  --->空指针异常! 我想在websocket里面使用service 层的接口,从中获取数据库的一些信息  , 使用 @Autowired 注 ...

  3. 05.python解析式与生成器表达式

    解析式和生成器表达式 列表解析式 列表解析式List Comprehension,也叫列表推导式 #生成一个列表,元素0-9,将每个元素加1后的平方值组成新的列表 x = [] for i in ra ...

  4. 使用.NET 6开发TodoList应用(30)——实现Docker打包和部署

    系列导航及源代码 使用.NET 6开发TodoList应用文章索引 需求 .NET 6 Web API应用使用最多的场景是作为后端微服务应用,在实际的项目中,我们一般都是通过将应用程序打包成docke ...

  5. Web开发之Servlet

    当一个请求到达服务端,服务器怎么处理? 当一个请求到达服务端时,由服务端的引擎来进行分析.它根据工程名找到工程, 然后拿到URL的资源地址和web.XML文件的所有的进行对比,和哪一个对比上就找到了具 ...

  6. linux中链接错误的时候,快速找到缺失的符号在哪个库中

    编译一个opencv程序,链接的时候出现大量的如下错误: /home/admin/opencv/opencv-master/modules/imgproc/src/color_lab.cpp:23: ...

  7. 搭建服务器之FTP

    FTP服务器,使用软件vsftpd,服务守护进程也是vsftpd.客户端访问的话可以用浏览器或ftp命令行. 1.yum install vsftpd.安装简单主要是配置,这个比httpd复杂点的地方 ...

  8. gin中绑定uri

    package main import ( "github.com/gin-gonic/gin" "net/http" ) type Person struct ...

  9. Jquery Ajax添加header参数

    在使用ajax请求接口时需要在请求头添加token来进行身份验证,方式如下: $.ajax({ type: 'GET', url: 'http://api.php', dataType: 'json' ...

  10. 测试开发实战[提测平台]20-图表G2Plot在项目的实践实录

    微信搜索[大奇测试开],关注这个坚持分享测试开发干货的家伙. G2Plot项目应用 上一篇<提测平台19-Echarts图表在项目的实践>讲解了Echarts的图表应用,此篇来看下开箱即用 ...