题意:

题解: 这道题我思路大方向是正确的,但是生成函数推错导致一直WA,看了标程才改对……

首先一个长为\(m\)的轮换的\(n\)次幂会分裂成\(\gcd(n,m)\)个长为\(\frac{m}{\gcd(n,m)}\)的轮换
所以合并的时候相当于对于一个长度\(l\)若存在一个\(m\)使得\(\frac{m}{\gcd(n,m)}=l\)则\(\gcd(n,m)\)个长度为\(l\)的轮换可以合并

显然不同长度的轮换是互不影响的,那么我们可以分开每种长度计算
就相当于对于一个长度为\(l\)的有标号的轮换,要把它们划分成若干无标号集合,每个集合大小都在给定的集合\(S\)内,并且每个集合有权值(合并的方案数),一种划分方案的权值为所有集合权值之积,求所有划分方案权值总和
那么显然这个东西的EGF就等于\(\exp(\sum_{i\in S} \frac{w_i}{i!})\), \(w_i\)为权值

如何求\(w_i\)? 在这里我出了问题
正确的答案是,假设\(k\)个长度为\(l\)的轮换合并,方案数为\(l^{k-1}(k-1)!\), 也就是\(\frac{l^kk!}{lk}\).
这大概是因为,假设我们定住第一个轮换的第一个元素的位置,那么其余每个轮换自身内部的顺序都可以改变,这样是\(l^{k-1}\)种;这些轮换之间的顺序也可以改变,这样是\((k-1)!\)种。

代码

#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cassert>
#include<iostream>
#include<vector>
#define llong long long
using namespace std;inline int read()
{int x=0; bool f=1; char c=getchar();for(;!isdigit(c);c=getchar()) if(c=='-') f=0;for(; isdigit(c);c=getchar()) x=(x<<3)+(x<<1)+(c^'0');if(f) return x;return -x;
}const int N = 1<<19;
const int LGN = 19;
const int P = 998244353;
const int G = 3;llong quickpow(llong x,llong y)
{llong cur = x,ret = 1ll;for(int i=0; y; i++){if(y&(1ll<<i)) {y-=(1ll<<i); ret = ret*cur%P;}cur = cur*cur%P;}return ret;
}
llong mulinv(llong x) {return quickpow(x,P-2);}namespace FFT
{llong tmp1[N+3],tmp2[N+3],tmp3[N+3],tmp4[N+3],tmp5[N+3],tmp6[N+3],tmp7[N+3],tmp8[N+3],tmp9[N+3],tmp10[N+3];llong tst1[N+3],tst2[N+3],tst3[N+3];llong sexp[N+3];int fftid[N+3];int getdgr(int n) {int ret = 1; while(ret<=n) ret<<=1; return ret;}void init_fftid(int dgr){int len = 0; for(int i=1; i<=LGN; i++) {if((1<<i)==dgr) {len = i; break;}}fftid[0] = 0; for(int i=1; i<dgr; i++) fftid[i] = (fftid[i>>1]>>1)|((i&1)<<(len-1));}void ntt(int dgr,int coe,llong poly[],llong ret[]){init_fftid(dgr);if(poly==ret) {for(int i=0; i<dgr; i++) {if(i<fftid[i]) swap(ret[i],ret[fftid[i]]);}}else {for(int i=0; i<dgr; i++) ret[i] = poly[fftid[i]];}for(int i=1; i<=(dgr>>1); i<<=1){llong tmp = quickpow(G,(P-1)/(i<<1));if(coe==-1) {tmp = mulinv(tmp);}sexp[0] = 1ll; for(int j=1; j<i; j++) sexp[j] = sexp[j-1]*tmp%P;for(int j=0; j<dgr; j+=(i<<1)){for(llong *k=ret+j,*kk=sexp; k<ret+i+j; k++,kk++){llong y = k[i]*(*kk)%P;k[i] = (*k)-y<0 ? (*k)-y+P : (*k)-y;(*k) = (*k)+y>=P ? (*k)+y-P : (*k)+y;}}}if(coe==-1){llong tmp = mulinv(dgr);for(int i=0; i<dgr; i++) ret[i] = ret[i]*tmp%P;}}void polymul(int dgr,llong poly1[],llong poly2[],llong ret[]){ntt((dgr<<1),1,poly1,tmp1); ntt((dgr<<1),1,poly2,tmp2);for(int i=0; i<(dgr<<1); i++) ret[i] = tmp1[i]*tmp2[i]%P;ntt((dgr<<1),-1,ret,ret);}void polyinv(int dgr,llong poly[],llong ret[]){for(int i=0; i<(dgr<<1); i++) ret[i] = tmp3[i] = tmp4[i] = tmp5[i] = 0ll;ret[0] = mulinv(poly[0]);for(int i=1; i<=(dgr>>1); i<<=1){for(int j=0; j<(i<<1); j++) tmp3[j] = poly[j];ntt((i<<2),1,tmp3,tmp4); ntt((i<<2),1,ret,tmp5);for(int j=0; j<(i<<2); j++) tmp3[j] = tmp4[j]*tmp5[j]%P*tmp5[j]%P;ntt((i<<2),-1,tmp3,tmp4);for(int j=0; j<(i<<1); j++) ret[j] = (ret[j]+ret[j]-tmp4[j]+P)%P;}}void polyder(int dgr,llong poly[],llong ret[]){for(int i=0; i<dgr-1; i++) ret[i] = poly[i+1]*(i+1)%P;ret[dgr-1] = 0ll;}void polyint(int dgr,llong poly[],llong ret[]){for(int i=1; i<dgr; i++) ret[i] = poly[i-1]*mulinv(i)%P;ret[0] = 0ll;}void polyln(int dgr,llong poly[],llong ret[]){for(int i=0; i<(dgr<<1); i++) ret[i] = tmp6[i] = tmp7[i] = tmp8[i] = 0ll;polyder(dgr,poly,tmp6);polyinv(dgr,poly,tmp7);polymul(dgr,tmp6,tmp7,tmp8);polyint(dgr,tmp8,ret);}void polyexp(int dgr,llong poly[],llong ret[]){for(int i=0; i<(dgr<<1); i++) ret[i] = tmp9[i] = tmp10[i] = 0ll;ret[0] = 1ll;for(int i=1; i<=(dgr>>1); i<<=1){polyln((i<<1),ret,tmp9);for(int j=0; j<(i<<1); j++) tmp9[j] = (-tmp9[j]+poly[j]+P)%P; tmp9[0]++;polymul((i<<2),ret,tmp9,tmp10);for(int j=0; j<(i<<1); j++) ret[j] = tmp10[j];}}
}
int permu[N+3];
int a[N+3];
int dgr[N+3];
bool vis[N+3];
vector<int> s[N+3];
llong f[N+3],expf[N+3];
llong fact[N+3];
int n;int gcd(int x,int y) {return y==0?x:gcd(y,x%y);}int main()
{fact[0] = 1ll; for(int i=1; i<=N; i++) fact[i] = fact[i-1]*i%P;scanf("%d",&n);for(int i=1; i<=n; i++) scanf("%d",&permu[i]);for(int i=1; i<=n; i++){if(vis[i]) continue;int len = 1; vis[i] = true;for(int j=permu[i]; j!=i; j=permu[j]){len++;vis[j] = true;}a[len]++;}for(int i=1; i<=n; i++){int g = gcd(i,n),l = i/gcd(i,n);s[l].push_back(g);}llong ans = 1ll;for(int i=1; i<=n; i++){if(a[i]==0) continue;int dgr = FFT::getdgr(a[i]);for(int j=0; j<s[i].size(); j++){if(s[i][j]<dgr){f[s[i][j]] = (f[s[i][j]]+mulinv(s[i][j])*quickpow(i,s[i][j]-1))%P;}}FFT::polyexp(dgr,f,expf);ans = ans*expf[a[i]]%P*fact[a[i]]%P;for(int j=0; j<(dgr<<1); j++) f[j] = expf[j] = 0ll;}printf("%lld\n",ans);return 0;
}

Luogu P4709 信息传递 (群论、生成函数、多项式指数函数)相关推荐

  1. luogu P4726 多项式指数函数(模板题FFT、多项式求逆、多项式对数函数)

    luogu P4726 多项式指数函数(模板题FFT.多项式求逆.多项式对数函数) 手动博客搬家: 本文发表于20181127 08:39:42, 原地址https://blog.csdn.net/s ...

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

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

  3. 2015 NOIP day2 t2 信息传递 tarjan

    信息传递 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://www.luogu.org/problem/show?pid=2661 Descrip ...

  4. 从信息传递的视角看因果关系

    来源:混沌巡洋舰 因果推断让大数据更有温度在六种看待因果关系的视角(机制,反事实,干预,过程,信息传递及概率因果)中,信息传递是潜在应用最广泛的一种.当我们想弄清楚是什么导致某件事情发生时,如果由于伦 ...

  5. AC日记——信息传递 洛谷 P2661 (tarjan求环)

    题目描述 有n个同学(编号为1到n)正在玩一个信息传递的游戏.在游戏里每人都有一个固定的信息传递对象,其中,编号为i的同学的信息传递对象是编号为Ti同学. 游戏开始时,每人都只知道自己的生日.之后每一 ...

  6. [拓扑排序/强联通分量] [NOIP201402] 信息传递

    信息传递 题目描述 有n个同学(编号为1到n)正在玩一个信息传递的游戏.在游戏里每人都有一个固定的信息传递对象,其中,编号为i的同学的信息传递对象是编号为Ti同学. 游戏开始时,每人都只知道自己的生日 ...

  7. [洛谷P4726]【模板】多项式指数函数

    题目大意:给出$n-1$次多项式$A(x)$,求一个 $\bmod{x^n}$下的多项式$B(x)$,满足$B(x) \equiv e^{A(x)}$. 题解:(by Weng_weijie) 泰勒展 ...

  8. 2019年东莞特长生 游戏(洛谷 P2661 信息传递)

    Description 某校科技节到了,有? 个同学(编号为1到?)正在玩一个信息传递的游戏.在游戏里每人都有一个固定的信息传递对象,其中,编号为? 的同学的信息传递对象是编号为??的同学. 游戏开始 ...

  9. tg2015 信息传递 (洛谷p2661)

    题目描述 有n个同学(编号为1到n)正在玩一个信息传递的游戏.在游戏里每人都有一个固定的信息传递对象,其中,编号为i的同学的信息传递对象是编号为Ti同学. 游戏开始时,每人都只知道自己的生日.之后每一 ...

最新文章

  1. 在析构函数中delete this指针问题
  2. CentOS 6.6安装Xtrabackup RPM提示缺少libev.so.4()
  3. 号外! 大维度机器学习也有计算框架了
  4. css阻止input select默认事件
  5. Java将一段逗号分割的字符串转换成一个数组(亲测)
  6. HDU3662(求三维凸包表面的多边形个数,表面三角形个数,体积,表面积,凸包重心,凸包中点到面的距离)
  7. OA项目实战(二) 开发准备
  8. 数学建模中常用的方法
  9. Databricks文档02----Databricks CLI
  10. 乐视网:对FF与第九城市设立合资公司的计划不知情
  11. OpenShift 4 - Knative教程 (6) Eventing之Channel和Subscription
  12. 局部页面切换url为什么不变_python爬虫 - 翻页url不变网页的爬虫探究!
  13. 我的第二个切换图片高亮显示,给力
  14. python图片表格转excel表格_python提取图片内容并转换成对应表格的markdown代码
  15. 如何判断自己外网IP是否为真实公网IP,以及解决方案
  16. python取省边界_提取行政区边界经纬度坐标(高德+百度)
  17. 电源管理芯片LDO(Low Dropout Regulator)分析1
  18. vantUI弹出框和微信小程序的究极大坑
  19. 不可不知!4种常见的黑客攻击手段
  20. 阿里巴巴直播防控中的实人认证技术 1

热门文章

  1. wpf cefsharp html源码,CefSharp For WPF基本使用
  2. Linux下修改命令提示符
  3. C#中代理的简单应用
  4. 需求分析的20条法则
  5. javascript闭包产生的内存泄漏
  6. EffectiveC++编程的50个建议
  7. XCTF-Reverse:insanity
  8. Java 接口interface
  9. app包中的fragment和v4包中的fragment的使用的区别
  10. idea debug只断点当前线程,不阻塞其他线程