排列与置换换+容斥+多项式生成函数启发式合并:[Gym-103446B]

news/2025/3/14 20:36:19/

https://vjudge.net/contest/591700#problem/G

看到排列,先考虑置换换,题意转化为置换环相邻的不能再最终序列上相邻

而这个过程看起来很容斥,所以我们容斥:至少要 x x x 个相邻

我们发现每个置换环的所有边不能全部同时被选,所以我们每个置换环要分开考虑,最后再乘起来

然而这样的复杂度会炸。但是我们可以写成生成函数的形式,然后用启发式合并起来

#include<bits/stdc++.h>
using namespace std;
#ifdef LOCAL#define debug(...) fprintf(stdout, ##__VA_ARGS__)
#else#define debug(...) void(0)
#endif
#define int long long
inline int read(){int x=0,f=1;char ch=getchar(); while(ch<'0'||
ch>'9'){if(ch=='-')f=-1;ch=getchar();}while(ch>='0'&&ch<='9'){
x=(x<<1)+(x<<3)+(ch^48);ch=getchar();}return x*f;}
#define Z(x) (x)*(x)
#define pb push_back
//srand(time(0));
#define N 2000010
//#define M
#define mo 998244353
#define G 3
inline int pw(int a, int b) {int ans=1; while(b) {if(b&1) ans*=a; a*=a; b>>=1; ans%=mo; a%=mo; }return ans; 
}
int fac[N], ifac[N]; 
void init(int n) {int i; for(i=fac[0]=1; i<=n; ++i) fac[i]=fac[i-1]*i%mo; ifac[n]=pw(fac[n], mo-2); for(i=n-1; i>=0; --i) ifac[i]=ifac[i+1]*(i+1)%mo; 
}
int C(int n, int m) {if(m>n) return 0;return fac[n]*ifac[m]%mo*ifac[n-m]%mo; 
}
const int Gi=pw(G, mo-2); 
int Mod(int a) { if(a>=mo || a<=-mo) a%=mo; if(a<0) a+=mo; return a; }
void Add(int &a, int b) { a+=b; Mod(a); }
void Mul(int &a, int b) { Mod(b); a*=b; Mod(a); } 
int rev[1<<22]; 
inline void revese(int n, int l) {for(int i=0; i<n; ++i) rev[i]=((rev[i>>1]>>1)|((i&1)<<l-1)); 
}
inline void NTT(int *P, int n, int op) {int i, j, k, w, W, X, Y; for(i=0; i<n; ++i) if(i<rev[i]) swap(P[i], P[rev[i]]); for(i=1; i<n; i<<=1) {W=pw(op==1 ? G : Gi, (mo-1)/(i<<1)); //*****for(j=0; j<n; j+=(i<<1)) {for(k=0, w=1; k<i; ++k, w=Mod(w*W)) {X=P[j+k], Y=Mod(w*P[j+k+i]); P[j+k]=Mod(X+Y), P[j+k+i]=Mod(X-Y); }}}if(op==1) return ; int inv=pw(n, mo-2); for(i=0; i<n; ++i) P[i]=P[i]*inv%mo; 
}
struct node {int x, len; bool operator <(const node &A) const {return len>A.len; }
};
priority_queue<node>q; 
int u, ve; 
int n, m, i, j, k, T;
int f[N], g[N], rt; 
int a[N]; 
int F[N], w[N], ans; 
vector<int>v[N]; int fa(int x) {if(F[x]==x) return x; return F[x]=fa(F[x]); 
}signed main()
{#ifdef LOCALfreopen("in.txt", "r", stdin);freopen("out.txt", "w", stdout);#endif
//	T=read();
//	while(T--) {
//
//	}srand(time(0)); m=read();  init(m); for(i=1; i<=m; ++i) F[i]=i; for(i=1; i<=m; ++i) k=read(), F[fa(i)]=fa(k); for(i=1; i<=m; ++i) w[fa(i)]++; for(i=1; i<=m; ++i) if(fa(i)==i) {debug("%lld %lld\n", i, w[i]); ++n; for(j=0; j<w[i]; ++j) v[n].pb(C(w[i], j)); v[n].pb(0); q.push({n, w[i]+1}); }while(!q.empty()) {u=q.top().x; q.pop(); if(q.empty()) break; ve=q.top().x; q.pop(); k=++n; int len1=v[u].size()-1, len2=v[ve].size()-1; int len=len1+len2+1, m=len, n, le; //new n!for(n=1, le=0; n<=m; n<<=1) ++le; revese(n, le);  for(i=0; i<v[u].size() && i<n; ++i) f[i]=v[u][i]; for(; i<n; ++i) f[i]=0; for(i=0; i<v[ve].size() && i<n; ++i) g[i]=v[ve][i]; for(; i<n; ++i) g[i]=0; for(i=0; i<n; ++i) debug("%lld ", f[i]); debug("\n"); for(i=0; i<n; ++i) debug("%lld ", g[i]); debug("\n"); NTT(f, n, 1); NTT(g, n, 1); for(i=0; i<n; ++i) f[i]=Mod(f[i]*g[i]); NTT(f, n, -1); for(i=0; i<n; ++i) v[k].pb(f[i]); q.push({k, n}); }debug("%lld\n", n); for(i=0; i<m; ++i) debug("%lld ", v[n][i]); debug("\n"); for(i=0, k=1; i<m; ++i, k=-k) Add(ans, v[n][i]*k%mo*fac[m-i]%mo); ans=Mod(ans); printf("%lld\n", ans); return 0;
}

http://www.ppmy.cn/news/1199912.html

相关文章

<微信小程序>《微信小程序开发笔记》(二)

《微信小程序开发笔记》&#xff08;二&#xff09; 1 程序开发1.1 原则&#xff08;自己感悟&#xff09;1.2 架构1.3 开发模式 2 建立项目3 微信代码构成3.1 JSON 配置文件3.2 WXML 模板文件3.3 WXSS 样式文件3.4 JS 脚本逻辑文件 1 程序开发 1.1 原则&#xff08;自己感悟&…

线代学习笔记-向量

numpy广播机制&#xff0c;自动增加维数 numpy中的array函数生成向量&#xff0c;&#xff08;&#xff09;是函数标配&#xff0c;&#xff08;&#xff09;下必有一个[]表示向量元素集合&#xff0c;第一层[]下的后每一个[]代表一行&#xff0c;没有这个[]&#xff0c;表示这…

维乐 Prevail Glide带你做破风王者,无阻前行!

对于自行车骑手来说&#xff0c;需要应对的问题有很多&#xff0c;其中最大的问题之一&#xff0c;就是「风阻」。风阻永远都是你越反抗越强&#xff0c;因此为了克服风阻的力量&#xff0c;时间久了&#xff0c;身体自然会造成一定程度的损伤。如何才能调整前行的步伐&#xf…

计算机网络第4章-IPv4

IPv4数据报格式 IPv4数据报格式如下图所示 其中&#xff0c;有如下的关键字段需要特别注意&#xff1a; 版本&#xff08;号&#xff09;&#xff1a; 版本字段共4比特&#xff0c;规定了数据报的IP协议版本。通过查看版本号吗&#xff0c;路由器能确定如何解释IP数据报的剩…

VMware安装CentOS最小化开发环境导引

目录 一、概要 二、介绍 三、下载 四、安装 4.1 创建虚拟机 4.2 安装CentOS 五、配置网卡 六、配置本地安装源 七、安装软件 7.1 gcc/g 7.2 C的atomic库 7.3 java 7.4 Cmake 7.5 MariaDB客户端&#xff08;兼容mysql&#xff09; 八、用户配置文件.bash_profile…

监控易:支持多种协议和设备,适应复杂多变的IT环境

在当今的数字化时代&#xff0c;IT环境越来越复杂多变&#xff0c;各种设备、系统、应用程序需要实时地进行监控和管理&#xff0c;以保证业务的正常运行和高效性能。然而&#xff0c;传统的监控方案往往无法满足这些需求&#xff0c;因为它们通常只支持有限的协议和设备类型&a…

AtCoder Beginner Contest 327 G. Many Good Tuple Problems(带标号二分图计数+有区别小球放入有区别盒子)

题目 一个长为n(n<30)的原始序列x&#xff0c;x[i]可以取值0或1 一个长为m(m<1e9)的点对序列(s,t)&#xff0c; s序列第i项和t的第i项&#xff0c;均可以取值[1,n]&#xff0c; 如果构造好s和t后&#xff0c;对任意都存在01序列x使得&#xff0c; 则称这个序列是合法…

【Linux】:Linux项目自动化构建工具——make/Makefile || Linux第一个小程序——进度条(简单版本)

在本章开始给大家分享一个图片 希望对你有帮助 在这里插入图片描述 &#x1f3c6;前言 在开始本章之前 我们需要回顾一下上节课的函数的动静态库的优缺点 动态库的优点&#xff1a; 比较节省资源&#xff08;这里说的资源不仅仅是磁盘资源 也包括网络资源 内存资源等等&#…