P4071 [SDOI2016]排列计数

题目描述

求有多少种长度为 n 的序列 A,满足以下条件:

1 ~ n 这 n 个数在序列中各出现了一次

若第 i 个数 A[i] 的值为 i,则称 i 是稳定的。序列恰好有 m 个数是稳定的

满足条件的序列可能很多,序列数对 \(10^9+7\) 取模。

输入格式

第一行一个数 T,表示有 T 组数据。

接下来 T 行,每行两个整数 n、m。

输出格式

输出 T 行,每行一个数,表示求出的序列数

输入输出样例

输入 #1

5

1 0

1 1

5 2

100 50

10000 5000

输出 #1

0

1

20

578028887

60695423

说明/提示

测试点 1 ~ 3: $ T = 1000\(,\)n \leq 8\(,\)m \leq 8$;

测试点 4 ~ 6: $ T = 1000\(,\)n \leq 12\(,\)m \leq 12$;

测试点 7 ~ 9: $ T = 1000\(,\)n \leq 100\(,\)m \leq 100$;

测试点 10 ~ 12:$ T = 1000\(,\)n \leq 1000\(,\)m \leq 1000$;

测试点 13 ~ 14:$ T = 500000\(,\)n \leq 1000\(,\)m \leq 1000$;

测试点 15 ~ 20:$ T = 500000\(,\)n \leq 1000000\(,\)m \leq 1000000$。

【思路】

组合数学 / 错排

【题目大意】

a[i]在i的位置上面是稳定的

求有m个数是稳定的序列有多少个

【题目分析】

求符合条件序列的个数

符合条件序列可以先只看那稳定的数

一个序列中只有m个数是稳定的

其他的都是稳定的

那么稳定的数组合方式就是n个数里面取m个

因为如果稳定那一个数只对应一个位置

所以不存在顺序这一说,所以就是 \(C_n^m\)

知道了稳定数的组合方式

拿在看看除了这些稳定数之外数的组合方式

其他的数都是不在自己的位置上面的

也就是错排

直接用错排求出n-m(这里是减号下同下下同)个人错排方法的数量就好了

因为一种稳定数对应n-m个人的错排方式

所以数量数就是 \(C_n^m\) * (n-m)个人的错排方式

【存在的问题】

1.因为n和m的数据范围都是小于等于1e6

不是很小,而且T很大,

所以每次T 不能都单独求C和错排次数了

这样一定会超时

2.因为这道题中有取模运算

而递推求组合数只能过2000所以用这个阶乘求组合数就是必然的了

那么就要用到除法

而取模运算中不能用除法

【优化】

1.针对问题1

多次求解会超时

那就先预处理出来所有的阶乘和i个人的错排方式

到时候O(1)查询就好了

2.针对问题2

既然不能用除法那就用乘法

用逆元来代替除法就可以啦

求逆元因为模数是质数

所以可以用费马小定理求

费马小定理求逆元详见

这里

【完整代码】

#include<iostream>
#include<cstdio>
#define int long long
using namespace std;
const int k = 1e9 + 7;
const int Max = 1000005;
int d[Max];
int f[Max];
int inv[Max]; int p(int a,int b)
{
int ans = 1;
while(b)
{
if(b & 1)
ans = ans * a % k;
b >>= 1;
a = a * a % k;
}
return ans;
} signed main()
{
d[0] = 1,d[2] = 1;
for(register int i = 3;i <= 1000000;++ i)
d[i] = (i - 1) * (d[i - 1] + d[i - 2]) % k;
f[0] = 1,f[1] = 1;
for(register int i = 2;i <= 1000000;++ i)
f[i] = f[i - 1] * i % k;
inv[0] = p(f[0],k - 2);
for(register int i = 1;i <= 1000000;++ i)
inv[i] = p(f[i],k - 2) % k;
int t;
cin >> t;
while(t --)
{
int n,m;
scanf("%lld%lld",&n,&m);
//c(n,m) * d(n-m)
//n!/m!(n - m)! * d[n-m]
printf("%lld\n",(f[n] * inv[m] % k * inv[n-m] % k * d[n-m]) % k);
}
return 0;
}

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

随机推荐

  1. 打印从1到n位数的最大值

    题目: 输入数字n,按顺序打印从1到最大的n位十进制数,如输入3,则打印从1.2.3一直到最大的3位数999 参考大数运算的方法.考虑到位数会很大,所以采用字符串的形式解决.对输入的n,创建一个长度为 ...

  2. 金融finaunce单词finaunce财经

    金融(FINANCE或FINAUNCE)就是对现有资源进行重新整合之后,实现价值和利润的等效流通.(专业的说法是:实行从储蓄到投资的过程,狭义的可以理解为金融是动态的货币经济学.) 金融是人们在不确定 ...

  3. scrapy 爬虫中间件 deepth深度

    源码 class DepthMiddleware(object): def __init__(self, maxdepth, stats, verbose_stats=False, prio=1): ...

  4. sublime text3常用的一些快捷键

    --------------------------------下面的内容可以打印出来贴在电脑旁提醒自己-------------------- Ctrl + Shift + D  快速复制当前的一行 ...

  5. 19,flask消息闪现-flash

    Flash消息 请求完成后给用户的提醒消息,flask的核心特性, flash函数实现效果 视图函数中调用flash()方法 html中要使用get_flashed_messages() 后端代码: ...

  6. angularcli 第六篇(todolist 列表)

    1.通过文本框输入,向数组添加数据 <!-- 通过文本框输入,向数组添加数据 push --> <input type="text" name="111 ...

  7. Gtest:Using visual studio 2017 cross platform feature to compile code remotely

    参考:使用Visual Studio 2017作为Linux C++开发工具 前言 最近在学Gtest单元测试框架,由于平时都是使用Source Insight写代码,遇到问题自己还是要到Linux下 ...

  8. Kali下的内网劫持(一)

    ettercap利用计算机在局域网内进行通信的ARP协议的缺陷进行攻击,在目标主机与服务器之间充当中间人,嗅探两者之间的数据流量,从中窃取用户的数据信息,那么接下来我就给大家演示一下客户端的图片是怎么 ...

  9. NGINX状态模块的使用

    nginx状态模块可以用来查看当前nginx服务器的并发量和总的请求数 启用nginx的状态模块 状态模块需要在编译安装的时候启用. 1.下载nginx源码包 2.安装nginx并启用模块 3.修改n ...

  10. 类型擦除对Java调用Kotlin的影响

    @JvmName: 扩展方法相关: 先来定义一个扩展方法: 好,接下来再来定义一个扩展函数: 此时报错了..看一下错误提示: 其中给的提示有点奇怪,第一个是很明显咱们的扩展函数木有接收参数嘛,为啥提示 ...