洛谷题面传送门

废了,又不会做/ll

orz czx 写的什么神仙题解,根本看不懂(%%%%%%%%%

首先显然一个排列的贡献为其所有置换环的乘积。考虑如何算之。

碰到很多数的 LCM 之积只有两种可能,一是 Min-Max 容斥将 LCM 转化为 GCD,而是枚举质因子及其次数算贡献。但对于此题而言前者不是太可做(可能有复杂度不错(大概 \(n^2d(n)\)?)的解法,不过我没有细想所以也不太清楚),因此考虑后者。

考虑用类似于差分的思想,对于每个质因子 \(p\) 的每个次数 \(k\),我们考虑计算一下存在一个置换环大小为 \(p^k\) 的排列个数 \(c\),那么我们令答案乘以 \(p^c\),不难发现对于一个排列,如果其置换环大小中 \(p\) 次数最多的一项的次数为 \(k'\),那么它的贡献会在 \(k=1,2,3,\cdots,k'\) 处各被算一次,因此这样计算恰好可以算到所有置换环的贡献。于是限制问题转化为,如何求有多少个排列,满足其至少有一个置换环大小为 \(x=p^k\) 的倍数。

考虑从反面入手,即计算有多少个长度为 \(n\) 的排列不含任何长度为 \(x\) 的置换环,设其为 \(f_n\)。那么这个子问题的答案就是 \(n!-f_n\)。那么如何求 \(f_i\) 呢?我们再从反面入手,计算一下含有长度为 \(x\) 的置换环的排列个数(为什么要一来一回搞两次反面呢?因为直接做递推式中需要用到 \(f_i\)),那么我们枚举长度为 \(x\) 的置换环的长度之和 \(j\),设 \(g_j\) 表示有多少个长度为 \(j\) 的排列满足其每个置换环大小都是 \(x\) 的倍数,那么有 \(f_i=i!·\sum\limits_{j\in[1,i]}\dbinom{i}{j}g_jf_{i-j}\),组合数表示分配 \(j\) 个数给大小是 \(x\) 的倍数的置换环的方案数,一目了然。接下来考虑如何求 \(g_i\),我们枚举 \(1\) 所在的置换环的大小 \(j\),类似地有 \(\sum\limits_{j\in[1,i]}g_{i-j}\dbinom{i-1}{j-1}(j-1)!\)。注意到只有 \(x\mid i\) 时 \(g_i\ne 0\),因此有用的 \(f,g\) 各只有 \(\dfrac{n}{x}\) 个,也就保证了复杂度。这部分可能可以容斥,不过没有细想(

总复杂度 \(\sum\limits_{i=1}^n\dfrac{n^2}{i^2}=\mathcal O(n^2)\),因为 \(\sum\limits_{n>0}\dfrac{1}{n^2}<2\)。

const int MAXN=7500;
int n,mod,phi,c[MAXN+5][MAXN+5],fac[MAXN+5];
int qpow(int x,int e){
int ret=1;
for(;e;e>>=1,x=1ll*x*x%mod) if(e&1) ret=1ll*ret*x%mod;
return ret;
}
int pr[MAXN/5+5],prcnt=0,vis[MAXN+5];
void sieve(int n){
for(int i=2;i<=n;i++){
if(!vis[i]) pr[++prcnt]=i;
for(int j=1;j<=prcnt&&pr[j]*i<=n;j++){
vis[pr[j]*i]=1;if(i%pr[j]==0) break;
}
}
}
int f[MAXN+5],g[MAXN+5];
int calc(int x){
memset(f,0,sizeof(f));memset(g,0,sizeof(g));g[0]=1;
for(int i=x;i<=n;i+=x) for(int j=x;j<=i;j+=x) g[i]=(g[i]+1ll*c[i-1][j-1]*g[i-j]%phi*fac[j-1])%phi;
// for(int i=x;i<=n;i+=x) printf("%d %d\n",i,g[i]);
for(int i=n%x;i<=n;i+=x){
f[i]=fac[i];
for(int j=x;j<=i;j+=x) f[i]=(f[i]-1ll*f[i-j]*g[j]%phi*c[i][j]%phi+phi)%phi;
} return (fac[n]-f[n]+phi)%phi;
}
int main(){
scanf("%d%d",&n,&mod);phi=mod-1;sieve(n);
for(int i=(fac[0]=1);i<=n;i++) fac[i]=1ll*fac[i-1]*i%phi;
for(int i=0;i<=n;i++){
c[i][0]=1;
for(int j=1;j<=i;j++) c[i][j]=(c[i-1][j]+c[i-1][j-1])%phi;
} int res=1;
// for(int i=1;i<=n;i++) printf("%d\n",calc(i));
for(int i=1;i<=prcnt;i++) for(int j=pr[i];j<=n;j*=pr[i])
res=1ll*res*qpow(pr[i],calc(j))%mod;
printf("%d\n",res);
return 0;
}

洛谷 P6276 - [USACO20OPEN]Exercise P(组合数学+DP)的更多相关文章

  1. 洛谷P3158 [CQOI2011]放棋子 组合数学+DP

    题意:在一个m行n列的棋盘里放一些彩色的棋子,使得每个格子最多放一个棋子,且不同颜色的棋子不能在同一行或者同一列.有多少祌方法? 解法:这道题不会做,太菜了qwq.题解是看洛谷大佬的. 设C是组合数, ...

  2. 洛谷P2606 [ZJOI2010]排列计数 组合数学+DP

    题意:称一个1,2,...,N的排列P1,P2...,Pn是Magic的,当且仅当2<=i<=N时,Pi>Pi/2. 计算1,2,...N的排列中有多少是Magic的,答案可能很大, ...

  3. 洛谷P2507 [SCOI2008]配对 题解(dp+贪心)

    洛谷P2507 [SCOI2008]配对 题解(dp+贪心) 标签:题解 阅读体验:https://zybuluo.com/Junlier/note/1299251 链接题目地址:洛谷P2507 [S ...

  4. 洛谷 P3177 [HAOI2015]树上染色 树形DP

    洛谷 P3177 [HAOI2015]树上染色 树形DP 题目描述 有一棵点数为 \(n\) 的树,树边有边权.给你一个在 \(0 \sim n\)之内的正整数 \(k\) ,你要在这棵树中选择 \( ...

  5. 洛谷 P4072 [SDOI2016]征途 斜率优化DP

    洛谷 P4072 [SDOI2016]征途 斜率优化DP 题目描述 \(Pine\) 开始了从 \(S\) 地到 \(T\) 地的征途. 从\(S\)地到\(T\)地的路可以划分成 \(n\) 段,相 ...

  6. 洛谷P1880 石子合并(区间DP)(环形DP)

    To 洛谷.1880 石子合并 题目描述 在一个园形操场的四周摆放N堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆合并成新的一堆,并将新的一堆的石子数,记为该次合并的得分. 试设计出1 ...

  7. 洛谷P1063 能量项链(区间DP)(环形DP)

    To 洛谷.1063 能量项链 题目描述 在Mars星球上,每个Mars人都随身佩带着一串能量项链.在项链上有N颗能量珠.能量珠是一颗有头标记与尾标记的珠子,这些标记对应着某个正整数.并且,对于相邻的 ...

  8. 洛谷P1282 多米诺骨牌 (DP)

    洛谷P1282 多米诺骨牌 题目描述 多米诺骨牌有上下2个方块组成,每个方块中有1~6个点.现有排成行的 上方块中点数之和记为S1,下方块中点数之和记为S2,它们的差为|S1-S2|.例如在图8-1中 ...

  9. 洛谷 P5469 - [NOI2019] 机器人(区间 dp+拉格朗日插值)

    洛谷题面传送门 神仙题,放在 D1T2 可能略难了一点( 首先显然对于 P 型机器人而言,将它放在 \(i\) 之后它会走到左边第一个严格 \(>a_i\) 的位置,对于 Q 型机器人而言,将它 ...

随机推荐

  1. CQL和SQL的CRUD操作比较

    数据进行CRUD操作时,CQL语句和SQL语句的异同之处. 1.建表 2.CRUD语句比较 3.总结 1.建表 在此之前先分别创建两张表,插入数据,用来测试然后进行比较 在SQL数据库里面创建表 在C ...

  2. Mybatis 动态批量修改

    封面:学校夜景 xdm,祝大家节日快乐!! 今天听<路过人间>演唱会Live限定版,爱上了一句歌词. 说来惭愧,人对爱只学会,视死如归. 1.业务需求 如下: 前台传给我一个 docume ...

  3. Java中的函数式编程(八)流Stream并行编程

    写在前面 在本系列文章的第一篇,我们提到了函数式编程的优点之一是"易于并发编程". Java作为一个多线程的语言,它通过 Stream 来提供了并发编程的便利性. 题外话: 严格来 ...

  4. Noip模拟31 2021.8.5

    T1 Game 当时先胡了一发$\textit{Next Permutation}$... 然后想正解,只想到贪心能求最大得分,然后就不会了.. 然后就甩个二十分的走了... 正解的最大得分(叫它$k ...

  5. JVM:Java内存区域与内存溢出异常

    Java 虚拟机在执行 Java 程序的过程中会把它所管理的内存划分为若干个不同的数据区域.这些区域都有各自的用途,以及创建和销毁时间,有些区域随着虚拟机进程的启动而存在,有些区域依赖用户线程的启动和 ...

  6. Python学习笔记总结

    目录 Python学习笔记总结 前言 安装 数据类型 Hello,World 变量 字符串 首字母大写 全部小写 全部大写 Tab和换行符 格式化 去除空格 List列表 列表增删改查排序 遍历列表 ...

  7. AndroidStudio中debug.keystore文件不存在解决办法

    Android项目丢失了debug.keystore,直接重新生存一个key. 在cmd下,进入C:\Users\Administrator\.android目录执行命令如下:  keytool -g ...

  8. Linux部署Apollo+.Net Core简单使用

    Apollo官方网站非常详细,以下只是本人学习过程的整理 一.概念 Apollo(阿波罗)是一款可靠的分布式配置管理中心,能够集中化管理应用不同环境.不同集群的配置,配置修改后能够实时推送到应用端,并 ...

  9. Vue&Element开发框架中增加工作流处理,工作流的各个管理页面的界面处理

    我在起前面的几篇随笔中,大概介绍了工作流的一些场景化处理,包括如何把具体业务表单组件化,并在查看和编辑界面中,动态加载组件内容,以及对于查看申请单的主页面,把审批.取消.发起会签.会签.批示分阅.阅办 ...

  10. MySQL中特别实用的几种SQL语句送给大家

    在写SQL时,经常灵活运用一些SQL语句编写的技巧,可以大大简化程序逻辑.减少程序与数据库的交互次数,有利于数据库高可用性,同时也能显得你的SQL很牛B,让同事们眼前一亮. 目录 实用的SQL 1.插 ...