【CF715E】Complete the Permutations(容斥,第一类斯特林数)
【CF715E】Complete the Permutations(容斥,第一类斯特林数)
题面
CF
洛谷
给定两个排列\(p,q\),但是其中有些位置未知,用\(0\)表示。
现在让你补全两个排列,定义两个排列\(p,q\)之间的距离为每次选择\(p\)中两个元素交换,使其变成\(q\)的最小次数。
求距离恰好为\([0,n-1]\)的填数方案数。
加强的题目在\(BZOJ\)上有,戳这里。
题解
看到这道题目就觉得无比熟悉。回头翻了翻发现果然是省队集训的时候的题目。。。
果然都是原题啊。。。
如果排列已知,发现交换的最小次数显然就是沿着置换交换,交换次数为\(n-\)置换个数。那么考虑从\(p_i\)连边连向\(q_i\),那么要求的就是环的个数。
显然成链的中间的\(x-x\)边全部可以直接丢掉,那么只需要考虑最终的开头和结尾就知道这条链到底是什么类型的了。
那么边有四种:已知-已知,已知-\(0\),\(0\)-已知,\(0\)-\(0\)。
对于已经成环的部分,我们显然不需要再考虑的。那么我们要做的就是把已经存在的链给合并成环,还要凭空用\(0-0\)边构造出一些环。
设有\(k\)条链,\(x-0\)和\(0-x\)是分开考虑的,设\(g_i\)表示至少有\(i\)个环。
\]
其中\(m\)是\(0-0\)边的数量。
这里拿\(x-0\)边举例。首先选择若干个\(x-0\)边出来拼成\(x\)个环,枚举使用的边的个数组合数计算方案,然后把他们拼成环,环排列个数即第一类斯特林数。多出来的边随意拼接,显然他们需要找一个后继,即把\(0\)位置和另外一条边给拼起来,无论是选择另外一个\(0-0\)边来拼或者选择一个\(x-0\)边来拼都是可行的,因为每条边也只会有一个前驱,所以选择方案数是下降幂。\(0-x\)边是同理的。
设\(f_x\)为\(0-x\)边计算出来的恰好的结果,\(g_x\)为\(x-0\)边计算出来的恰好的结果。两者卷积后,得到的是恰好形成了\(x\)环的结果。但是此时还多出了一些\(0-0\)链,它们形成环的方案数还是斯特林数。因此再将卷积的结果和斯特林数卷积。
最终因为\(0-0\)之间无顺序关系,所以还要乘上一个阶乘。
#include<iostream>
#include<cstdio>
using namespace std;
#define MAX 550
#define MOD 998244353
void add(int &x,int y){x+=y;if(x>=MOD)x-=MOD;}
inline int read()
{
int x=0;bool t=false;char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=true,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return t?-x:x;
}
int n,a[MAX],b[MAX],deg[MAX];
int f[MAX],g[MAX],h[MAX],ans[MAX];
int nt[MAX],cnt[4];
bool vis[MAX<<1];
int C[MAX][MAX],A[MAX][MAX],S[MAX][MAX];
void dfs(int u,int ff)
{
vis[u]=true;
if(nt[u])
{
if(!vis[nt[u]])dfs(nt[u],ff);
else cnt[2]+=1;
}
else
{
if(u>n&&ff>n)cnt[3]+=1;
else if(u<=n&&ff>n)cnt[0]+=1;
else if(u>n&&ff<=n)cnt[1]+=1;
}
}
void calc(int *f,int n)
{
for(int i=0;i<=n;++i)
for(int j=i;j<=n;++j)
add(f[i],1ll*C[n][j]*S[j][i]%MOD*A[n-j+cnt[3]][n-j]%MOD);
for(int i=0;i<=n;++i)
{
int t=0;
for(int j=i,d=1;j<=n;++j,d=MOD-d)
add(t,1ll*d*f[j]%MOD*C[j][i]%MOD);
f[i]=t;
}
}
int main()
{
n=read();
for(int i=1;i<=n;++i)a[i]=read();
for(int i=1;i<=n;++i)b[i]=read();
for(int i=1;i<=n+n;++i)vis[i]=true;
for(int i=1;i<=n;++i)
{
if(!a[i])a[i]=i+n;if(!b[i])b[i]=i+n;
vis[a[i]]=vis[b[i]]=false;
if(a[i]<=n||b[i]<=n)nt[a[i]]=b[i],++deg[b[i]];
}
for(int i=1;i<=n+n;++i)if(!vis[i]&&!deg[i])dfs(i,i);
for(int i=1;i<=n+n;++i)if(!vis[i])dfs(i,i);
C[0][0]=S[0][0]=A[0][0]=1;
for(int i=1;i<=n;++i)
{
A[i][0]=C[i][0]=1;
for(int j=1;j<=i;++j)
{
C[i][j]=(C[i-1][j]+C[i-1][j-1])%MOD;
A[i][j]=1ll*A[i][j-1]*(i-j+1)%MOD;
S[i][j]=(S[i-1][j-1]+1ll*(i-1)*S[i-1][j])%MOD;
}
}
calc(f,cnt[0]);calc(g,cnt[1]);
for(int i=0;i<=n;++i)
for(int j=0;j<=i;++j)
add(h[i],1ll*f[j]*g[i-j]%MOD);
for(int i=0;i<=n;++i)
for(int j=0;j<=i;++j)
add(ans[i],1ll*h[j]*S[cnt[3]][i-j]%MOD);
for(int i=0;i<=n;++i)ans[i]=1ll*ans[i]*A[cnt[3]][cnt[3]]%MOD;
for(int i=0;i<n;++i)printf("%d ",n-i-cnt[2]>=0?ans[n-i-cnt[2]]:0);
puts("");return 0;
}
【CF715E】Complete the Permutations(容斥,第一类斯特林数)的更多相关文章
- UVA11077 Find the Permutations —— 置换、第一类斯特林数
题目链接:https://vjudge.net/problem/UVA-11077 题意: 问n的全排列中多有少个至少需要交换k次才能变成{1,2,3……n}. 题解: 1.根据过程的互逆性,可直接求 ...
- ARC096 E Everything on It [容斥,斯特林数]
Atcoder 一个900分的题耗了我这么久--而且官方题解还那么短--必须纪念一下-- 思路 发现每种元素必须出现两次以上的限制极为恶心,所以容斥,枚举出现0/1次的元素个数分别有几个.设出现1次的 ...
- CF715E Complete the Permutations(第一类斯特林数)
题目 CF715E Complete the Permutations 做法 先考虑无\(0\)排列的最小花费,其实就是沿着置换交换,花费:\(n-\)环个数,所以我们主要是要求出规定环的个数 考虑连 ...
- Codeforces 715E - Complete the Permutations(第一类斯特林数)
Codeforces 题面传送门 & 洛谷题面传送门 神仙题.在 AC 此题之前,此题已经在我的任务计划中躺了 5 个月的灰了. 首先考虑这个最短距离是什么东西,有点常识的人(大雾)应该知道, ...
- 【UVA 11077】 Find the Permutations (置换+第一类斯特林数)
Find the Permutations Sorting is one of the most used operations in real life, where Computer Scienc ...
- CF715E—— Complete the Permutations
传送门:QAQQAQ 题意:给你两个$1$~$n$的排列,0表示该位置数字不确定,两两交换第一个排列中的元素使之变成第二个排列,令$s[x]$表示对于所有不同的两个排列,最少交换次数为$x$的序列有$ ...
- 【HDU 4372】 Count the Buildings (第一类斯特林数)
Count the Buildings Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Othe ...
- 【组合数学:第一类斯特林数】【HDU3625】Examining the Rooms
Examining the Rooms Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Othe ...
- 如何快速求解第一类斯特林数--nlog^2n + nlogn
目录 参考资料 前言 暴力 nlog^2n的做法 nlogn的做法 代码 参考资料 百度百科 斯特林数 学习笔记-by zhouzhendong 前言 首先是因为这道题,才去研究了这个玩意:[2019 ...
随机推荐
- 关于LCA
LCA:最近公共祖先 指在有根树中,找出某两个结点u和v最近的公共祖先 如图,5,7的最近公共祖先就是3 接下来,我们来了解如何求解LCA No.1 暴力 首先想到的肯定是暴力,我们搜索,从两个节点一 ...
- hibernate 解决 java.lang.NoClassDefFoundError: Could not initialize class org.hibernate.validator.internal.engine.xxx 这类的问题
<!-- 解决 java.lang.NoClassDefFoundError: Could not initialize class org.hibernate.validator.intern ...
- Spring Boot(十六):使用 Jenkins 部署 Spring Boot
Jenkins 是 Devops 神器,本篇文章介绍如何安装和使用 Jenkins 部署 Spring Boot 项目 Jenkins 搭建.部署分为四个步骤: 第一步,Jenkins 安装 第二步, ...
- BodeAbp概述
BodeAbp框架基于github开源框架ASP.NET Boilerplate,abp项目地址:https://github.com/aspnetboilerplate/aspnetboilerpl ...
- 对于ps基本操作的归纳
1.开始新的制作 1)新建 快捷键:Ctrl+n 格式:宽高根据要求自选:颜色模式常用R(红)G(绿)B(蓝) 2)打开电脑上的图片 快捷键:Ctrl+o 2.选框工具 快捷键:M 作用:能 ...
- PHP中报500错误时如何查看错误信息
在执行代码中加入下面两行代码即可 ini_set("display_errors","On"); error_reporting(E_ALL);
- C_运算符_逻辑表达式
// 除法取余运算符的例子 //2018年9月19日22:44:21 # include<stdio.h> int main(void) { printf(%, %-, -%, -%-, ...
- swift 各种学习
swift使用cocoapods引用oc第三方库 1. 创建桥接文件 2. 在主工程的 build Settings 搜索 bridge 设置 Objective-C Bridging Headi ...
- springframework内BeanUtils源码使用记录一
package org.springframework.beans; public abstract class BeanUtils /** * Copy the property values of ...
- Excel读写方案XLSReadWriteII使用技巧总结
XLSReadWriteII是一个读写Excel的组件.他的一般已用只要按照Demo操作基本都能实现,只要不是非常复杂的应用,XLSReadWriteII还是能够胜任的. 最近被派了一个写入图库的应用 ...