SDOI2016-排列计数

发现很多题解都没有讲清楚这道题为什么要用逆元、递推公式怎么来的。 我,风雨兼程三十载,只为写出一篇好题解。 还是我来造福大家一下吧。

题目大意:

一个长度为 n 且 1~n 各出现一次的序列,希望在“序列中有且只有 m个数的值 等于 它的位置”条件下求出序列个数。答案对1000000007取模。

题目分析:

这道题也许是加强版的“装错了的信封”,在“装错了的信封”上搞搞比利就好。我们不妨设:

值等于位置的数字稳定的

值不等于位置数字不稳定的

稳定的数字由于要保证 值等于位置,故 稳定的数字从序列左到右的值是递增的

首先我们根据组合数的思想可以知道:

答案 = 稳定的挑选方法数 乘上 不稳定的挑选方法数

所以现在我们的目的改成了求出 稳定的挑选方法数不稳定的挑选方法数

稳定的挑选方法数:

我们已经知道了 稳定的数字从序列左到右的值是递增的,那么我们只需要模拟是哪几个数是稳定的即可。

不难想到,数量其实就是 在 \(n\) 个数中 挑选 \(m\) 个数的方案数

这不就是组合数里面的 \(C_n^m\) 吗?有 \(C_n^m\) = \(\frac{n!}{m!(n-m)!}\)

搞定

不稳定挑选方法数(错排):

我们假设已经知道了哪 \(m\) 个数是稳定的,那么我们可以在数组中暂时只看不稳定的数。

也就是 错排

我们令数组 \(F_x\) 为 有 \(x\) 个数字,每个数字均为不稳定的方法数

由于这些数是不稳定的,那么就没有值的大小递增关系,所以我们对于每一个 \(F_x\) ,都要进行分类讨论。

\[假设一个数字 k ,并令 n 位于第 k 位
\begin{cases}
当 k 位于第n位& \text{此时的错排数为 $F_{n-2}$(因 k 的位置已知,即求余的 n-2 个数的错排数)}\\
当 k 不位于第n位& \text{此时的错排数为 $F_{n-1}$}
\end{cases}\]

于是对于每个 \(F_i\) ,有 \(F_i\) = \((i - 1)\times(F_{x-1} + F_{x-2})\)

所以最终答案为:(\(C_n^m \times\)\(F_{n-m}\))%MOD

具体操作思路:

我们先要初始化,对于 \(C_x^y\) 要初始化阶乘,对于 \(F_i\) 要递推。

然后对于每一组数据,套 (\(C_n^m \times\)\(F_{n-m}\))%MOD 即可。

但是(还没完):

由题意得,最终方案数可能很大(所以才要取模),我们在进行 答案累乘时,(\(C_n^m \times\)\(F_{n-m}\))%MOD 可能会爆精度,我们于是要用到:

逆元(inv)

何为逆元( \(inv\) )?

比如当有 \((a \div b) \% MOD\) 时,防止 \(b\) 过大而爆精度,将 \(a \div b\) 转化为某种简单的乘法。

若 \(c\) 为 \(b\) 的逆元,则有 \((b \times c) \equiv 1 (mod 模数)\)

那么 \((a \div b) \% MOD\) = \((a \div b) \times 1 \% MOD\) = \((a \div b) \times b \times c \% MOD\) = \((a \times c) \% MOD\)

即 (一个数 除以 另一个数)%模数 = (一个数 乘上 另一个数的逆元)%模数

如何将逆元应用到该题中

我们初始化逆元数组 \(inv\) , \(inv\) 为 阶乘的逆元

那对于固定的值于模数,那个值的逆元怎么求呢?

请你参考费马小定理

直接给出答案:对于 \(a \% 1000000007\) 的逆元,为 \(a^{1000000007-2}\)

搞个快速幂就好了

代码:

#include<bits/stdc++.h>
#include<cctype>
#pragma GCC optimize(2)
#define Max(a,f) a > b ? a : b
#define Min(a,b) a < b ? a : b
#define in(n) n = read()
#define out(a) write(a)
#define outn(a) out(a),putchar('\n')
#define New ll
#define ll long long
#define rg register
using namespace std; namespace IO_Optimization//优化函数
{
inline New read()//快读
{
New X = 0,w = 0;
char ch = 0; while(!isdigit(ch))
{
w |= ch == '-';
ch=getchar();
}
while(isdigit(ch))
{
X = (X << 3) + (X << 1) + (ch ^ 48);
ch = getchar();
}
return w ? -X : X;
} inline void write(New x)//快输
{
if(x < 0) putchar('-'),x = -x;
if(x > 9) write(x / 10);
putchar(x % 10 + '0');
}
}
using namespace IO_Optimization; const int mod = 1000000000 + 7;//模数
const int MAXN = 1000000 + 2;//数组大小常量 ll a[MAXN],f[MAXN],inv[MAXN];
// 阶乘 F数组 逆元数组 inline ll ksm(ll a,ll b){//快速幂函数,求逆元
ll ans = 1;
a %= mod;
while(b)
{
if(b & 1)
ans = (a * ans) % mod;
a = (a * a) % mod;
b >>= 1;
}
return ans;
} int main()
{
a[0] = 1;//初始化
for(rg int i = 1;i <= MAXN; ++i)
{
a[i] = (a[i - 1] * i) % mod;//记得取模
inv[i] = ksm(a[i],mod - 2);//计算逆元
}
f[1] = 0,f[2] = 1,f[3] = 2;//初始化
for(rg int i = 4;i <= MAXN; ++i)
f[i] = ((i - 1) * (f[i - 1] + f[i - 2])) % mod;//递推公式
int T = read();//多数据输入
while(T--)
{
int n = read(),m = read(),k = n - m;//输入n、m,k纯属方便
if(!k)//如果n-m=0,那么方案数为1
{
puts("1");
continue;
}
if(m == 0)//如果没有稳定的数,那么答案直接等于F[n]
{
outn(f[n]);
continue;
}
if(k == 1)//如果n-m=1,则有0个方案
{
puts("0");
continue;
}
outn(((( a[n] * inv[k] ) % mod * inv[m]) % mod * f[k]) % mod);//输出
//上面是把除法改成乘法逆元了
}
return 0;
}

洛谷P4071-[SDOI2016]排列计数 题解的更多相关文章

  1. 洛谷 P4071 [SDOI2016]排列计数 题解

    P4071 [SDOI2016]排列计数 题目描述 求有多少种长度为 n 的序列 A,满足以下条件: 1 ~ n 这 n 个数在序列中各出现了一次 若第 i 个数 A[i] 的值为 i,则称 i 是稳 ...

  2. 洛谷——P4071 [SDOI2016]排列计数(错排+组合数学)

    P4071 [SDOI2016]排列计数 求有多少种长度为 n 的序列 A,满足以下条件: 1 ~ n 这 n 个数在序列中各出现了一次 若第 i 个数 A[i] 的值为 i,则称 i 是稳定的.序列 ...

  3. 洛谷P4071 [SDOI2016] 排列计数 [组合数学]

    题目传送门 排列计数 题目描述 求有多少种长度为 n 的序列 A,满足以下条件: 1 ~ n 这 n 个数在序列中各出现了一次 若第 i 个数 A[i] 的值为 i,则称 i 是稳定的.序列恰好有 m ...

  4. 洛谷 P4071 [SDOI2016]排列计数

    洛谷 这是一道组合数学题. 对于一个长为n的序列,首先我们要选m个使之稳定\(C^{m}_{n}\). 且要保证剩下的序列不稳定,即错排\(D_{n-m}\). 所以答案就是:\[ANS=C^{m}_ ...

  5. P4071 [SDOI2016]排列计数 题解

    分析: 线性求逆元:https://blog.csdn.net/qq_34564984/article/details/52292502 代码: #include<cstdio> usin ...

  6. 洛谷 P2606 [ZJOI2010]排列计数 解题报告

    P2606 [ZJOI2010]排列计数 题目描述 称一个\(1,2,...,N\)的排列\(P_1,P_2...,P_n\)是\(Magic\)的,当且仅当对所以的\(2<=i<=N\) ...

  7. BZOJ4517 & 洛谷4071:[SDOI2016]排列计数——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=4517 https://www.luogu.org/problemnew/show/P4071 求有 ...

  8. ●洛谷P2606 [ZJOI2010]排列计数

    题链: https://www.luogu.org/problemnew/show/P2606题解: 组合数(DP),Lucas定理 首先应该容易看出,这个排列其实是一个小顶堆. 然后我们可以考虑dp ...

  9. 洛谷P2606 [ZJOI2010]排列计数

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

  10. 洛谷P2606 [ZJOI2010]排列计数(数位dp)

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

随机推荐

  1. 四十一、在SAP中添加多条件选择框

    一.代码如下: 二.其中我们的文本替换内容如下 三.需要注意的是波浪线的用法,以及区域的添加方法.运行程序,显示如下 四.不勾选时,查询出来是去掉国际的 五.勾选之后,查询的是全部的 六.显示如下 七 ...

  2. 压测工具siege和wrk

    siege压测工具 安装: wget http://download.joedog.org/siege/siege-3.0.8.tar.gz cd siege-3.0.8 ./configure ma ...

  3. linux常用命令-关机、重启

    常用命令-关机.重启 命令 含义 reboot 重新启动操作系 shutdown –r now 重新启动操作系统,shutdown会给别的用户提示 shutdown -h now 立刻关机,其中now ...

  4. Markdown工具推荐

    Markdown 因语法简单,应用广泛,在近几年被很多开发者所喜爱.常用的语法不多,也就十来种吧.本人自从2017年接触就一发不可收拾. 在这几年里,用过了很多写Markdown的工具.接下来就以现在 ...

  5. Cobalt Strike简单使用(9,29第十五天)

    本文转自:https://www.cnblogs.com/yuanshu/p/11616657.html 一.介绍: 后渗透测试工具,基于Java开发,适用于团队间协同作战,简称“CS”. CS分为客 ...

  6. 【Vue中的坑】Vue中的@mouseenter没反应?

    在开发中想实现鼠标悬浮,然后发现事件不由被出发,查找资料,发现并不是所有情况都不能用 下面就简单的说一下如何避免这种情况 如果你的悬浮事件是在 a 标签上,那么你直接使用就会出问题,你需要加一个nat ...

  7. Day3-T2

    原题目 奶牛Bessie的电脑总是无缘无故地被 FJ 关掉,奶牛 Bessie 非常苦恼,也非常生气.FJ 却发现了 一个很神奇的规律(别问是怎么知道的),发现 Bessie 每吃一次草,她的生气值会 ...

  8. comm

    comm [- 123 ] file1 file2 说明:该命令是对两个已经排好序的文件进行比较.其中file1和file2是已排序的文件.comm读取这两个文件,然后生成三列输出:仅在file1中出 ...

  9. 一、CI框架(CodeIgniter)简介

    CI是一个非常好用,非常灵活的PHP框架,在官网https://codeigniter.org.cn/ :最新版本3.1.10 版 就可以尽情使用了. 不忘初心,如果您认为这篇文章有价值,认同作者的付 ...

  10. Linux - 安装 dotnet core 环境

    Linux -  安装 dotnet core 环境 系统环境:CentOS7 官方安装指导 https://www.microsoft.com/net/learn/get-started/linux ...