组合数取模的几个方法

求:

\[C^{m}_{n} \bmod P
\]

1.杨辉三角法

\[C^{m}_{n} = C^{m - 1}_{n - 1} + C^{ m }_{n - 1}
\]

时间复杂度有点小高

2.若 \(n\) 比较大,但 \(P\) 比较小

使用 \(EXlucas\) 大法.

具体讲解点这里:戳我

3.若 \(n\) 比较小, \(P\) 炒几大

我们考虑分解组合数:

原式为:

\[\frac{n!}{\left(n - m\right)! \times m! }
\]

化简为:

\[\frac{\prod_{i = m + 1}^{n}i}{\prod^{n - m}_{i=1}i}
\]

考虑对每一个 \(i\) 分解质因数,但由于全分解的时间复杂度忒高,于是先将其最小的分出来。

开一个长度为 \(n\) 的 \(cnt\) 数组 \(cnt_i\) 表示 \(i\) 的 \(cnt_i\) 次幂。

从大往小里枚举,只要其不是质数,就考虑如下转化:


cnt[i / num[i]] += cnt[i]; cnt[num[i]] += cnt[i];

易证其正确性。

下面一道例题: ( \(Catlan\) )

\(HNOI2009\) 有趣的数列

我们称一个长度为 \(2n\) 的数列是有趣的,当且仅当该数列满足以下三个条件:

  • 它是从 \(1 \sim 2n\) 共 \(2n\) 个整数的一个排列 \(\{a_n\}_{n=1}^{2n}\);

  • 所有的奇数项满足 \(a_1<a_3< \dots < a_{2n-1}\),所有的偶数项满足 \(a_2<a_4< \dots <a_{2n}\);

  • 任意相邻的两项 \(a_{2i-1}\) 与 \(a_{2i}\) 满足:\(a_{2i-1}<a_{2i}\)。

对于给定的 \(n\),请求出有多少个不同的长度为 \(2n\) 的有趣的数列。

因为最后的答案可能很大,所以只要求输出答案对 \(p\) 取模。

输入格式

一行两个正整数 \(n,p\)

输出格式

输出一行一个整数表示答案。

样例输入

3 10

样例输出

5

数据范围

对于 \(50\%\) 的数据,\(1\le n \le 1000\);

对于 \(100\%\) 的数据,\(1\le n \le 10^6\),\(1\le p \le 10^9\)。

\(code\)

简化题意,求:

\[\frac{C^{n}_{2n}}{n + 1}
\]

易推出是卡特兰数,代码如下:

114514
#include<bits/stdc++.h>
using namespace std ;
#define int long long
const int N = 1000010 ;
int n , mod ;
int num , prime[ 2 * N ] ;
int Next[ N << 1 ] ;
int cnt[ N << 1 ] ;
inline int read( )
{
int x = 0 , f = 1 ;
char c = getchar( ) ;
while ( c > '9' || c < '0' )
{
if( c == '-' )
{
f = -f ;
}
c = getchar( ) ;
}
while ( c >= '0' && c <= '9' )
{
x = x * 10 + c - '0' ;
c = getchar( ) ;
}
return x * f ;
}
inline int Regular_Quick_Pow( int a , int b )
{
int ans = 1 ;
while ( b > 0 )
{
if ( b & 1 ) ans = ( ans * a ) % mod ;
b >>= 1 ;
a = ( a * a ) % mod ;
}
return ans ;
}
signed main( )
{
#ifndef ONLINE_JUDGE
freopen( "1.in" , "r" , stdin ) ;
freopen( "1.out" , "w" , stdout ) ;
#endif
cin >> n >> mod ;
for ( int i = 2 ; i <= 2 * n ; ++ i )
{
if ( !Next[ i ] )
{
Next[ i ] = i ;
prime[ ++ num ] = i ;
}
for ( int j = 1 ; j <= num && prime[ j ] * i <= 2 * n ; ++ j )
{
Next[ prime[ j ] * i ] = min( prime[ j ] , Next[ i ] ) ;
}
}
Next[ 1 ] = 1 ;
for ( int i = n + 2 ; i <= 2 * n ; ++ i )
{
cnt[ i ] = 1 ;
}
for ( int i = 1 ; i <= n ; ++ i )
{
cnt[ i ] = -1 ;
}
for ( int i = 2 * n ; i >= 1 ; -- i )
{
if ( Next[ i ] != i )
{
cnt[ Next[ i ] ] += cnt[ i ] ;
cnt[ i / Next[ i ] ] += cnt[ i ] ;
cnt[ i ] = 0 ;
}
}
int ans = 1 ;
for ( int i = 1 ; i <= 2 * n ; ++ i )
{
ans = ( ans * Regular_Quick_Pow( i , cnt[ i ] ) ) % mod ;
}
cout << ans ;
}

组合数取模的几种方法--Exlucas&杨辉三角&组合的更多相关文章

  1. Java小题,通过JNI调用本地C++共享库中的对应方法实现杨辉三角的绘制

    1.在Eclipse中配置Javah,配置如下 位置是你javah.exe在你电脑磁盘上的路径 位置:C:\Program Files\Java\jdk1.8.0_112\bin\javah.exe ...

  2. Python练习题——用列表的方法输出杨辉三角

    def main(): num = int(input('请输入行数: ')) yh = [[]] * num #创建num行空列表 for row in range(len(yh)): #遍历每一行 ...

  3. java实现组合数_n!_杨辉三角_组合数递推公式_回文数_汉诺塔问题

    一,使用计算机计算组合数 1,设计思想 (1)使用组合数公式利用n!来计算Cn^k=n!/k!(n-k)!用递推计算阶乘 (2)使用递推的方法用杨辉三角计算Cn+1^k=Cn^k-1+Cn^k 通过数 ...

  4. 组合数取模Lucas定理及快速幂取模

    组合数取模就是求的值,根据,和的取值范围不同,采取的方法也不一样. 下面,我们来看常见的两种取值情况(m.n在64位整数型范围内) (1)  , 此时较简单,在O(n2)可承受的情况下组合数的计算可以 ...

  5. [BZOJ 3129] [Sdoi2013] 方程 【容斥+组合数取模+中国剩余定理】

    题目链接:BZOJ - 3129 题目分析 使用隔板法的思想,如果没有任何限制条件,那么方案数就是 C(m - 1, n - 1). 如果有一个限制条件是 xi >= Ai ,那么我们就可以将 ...

  6. BZOJ_2142_礼物_扩展lucas+组合数取模+CRT

    BZOJ_2142_礼物_扩展lucas+组合数取模 Description 一年一度的圣诞节快要来到了.每年的圣诞节小E都会收到许多礼物,当然他也会送出许多礼物.不同的人物在小E 心目中的重要性不同 ...

  7. 组合数取模&&Lucas定理题集

    题集链接: https://cn.vjudge.net/contest/231988 解题之前请先了解组合数取模和Lucas定理 A : FZU-2020  输出组合数C(n, m) mod p (1 ...

  8. HDU 5698 大组合数取模(逆元)

    瞬间移动 Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Submis ...

  9. Uva12034 (组合数取模)

    题意:两匹马比赛有三种比赛结果,n匹马比赛的所有可能结果总数 解法: 设答案是f[n],则假设第一名有i个人,有C(n,i)种可能,接下来还有f(n-i)种可能性,因此答案为 ΣC(n,i)f(n-i ...

  10. 组合数取模及Lucas定理

    引入: 组合数C(m,n)表示在m个不同的元素中取出n个元素(不要求有序),产生的方案数.定义式:C(m,n)=m!/(n!*(m-n)!)(并不会使用LaTex QAQ). 根据题目中对组合数的需要 ...

随机推荐

  1. .NET 认识日志系统-2

    .NET 日志系统2 上一篇文章是将日志打印到控制台,这篇文章将日志写入到文本文件中. 文本日志一般按照日期区分 如何避免文本日志把磁盘撑爆? 限制日志总个数或者总大小 如何避免一个日志文件太大? 限 ...

  2. Linux设备模型:5、device和device driver

    作者:wowo 发布于:2014-4-2 19:28 分类:统一设备模型 http://www.wowotech.net/device_model/device_and_driver.html 前言 ...

  3. 使用bootstrap-select 动态加载数据不显示的问题,级联数据置为空

    动态加载数据 $.showLoading('数据加载中');//开启遮挡层 $.ajax({ url: "/PickoutStock/GetSendReceive", data: ...

  4. 关于ComfyUI的一些Tips

    关于ComfyUI的一些Tips 前言: 最近发的ComfyUI相关文章节奏不知道会不会很快,在创作的时候没有考虑很多,想着把自己的知识分享出去.后台也看到很多私信,有各种各样的问题,这是我欠缺考虑了 ...

  5. 内网穿透的高性能的反向代理应用FRP-自定义404错误页【实践可行版】

    frp简介 frp 是一个专注于内网穿透的高性能的反向代理应用,支持 TCP.UDP.HTTP.HTTPS 等多种协议.可以将内网服务以安全.便捷的方式通过具有公网 IP 节点的中转暴露到公网. 为什 ...

  6. FFmpeg开发笔记(三十八)APP如何访问SRS推流的RTMP直播地址

    ​<FFmpeg开发实战:从零基础到短视频上线>一书在第10章介绍了轻量级流媒体服务器MediaMTX,通过该工具可以测试RTSP/RTMP等流媒体协议的推拉流.不过MediaMTX的功能 ...

  7. 10 pdf分享失败

    PC端分享pdf,复制粘贴pdf链接后跳转搜索首页

  8. Vue禁止用户复制文案 + 兼容 IE

    vue必须要加载完才可以操作dom,或者在mounted和created时使用this.$nextTick方法,使dom生成后进行相关操作. created() { this.$nextTick(() ...

  9. Nginx 工作原理简介

    在了解Nginx工作原理之前,我们先来了解下几个基本的概念 以及常见的I/O模型. 基本概念 同步:就是指调用方发起一个调用,在没有得到调用结果之前,该调用不返回.换句话说,也就是调用方发起一个调用后 ...

  10. Golang 高性能 Websocket 库 gws 使用与设计(一)

    前言 大家好这里是,白泽,这期分析一下 golang 开源高性能 websocket 库 gws. 视频讲解请关注B站:白泽talk 介绍 gws:https://github.com/lxzan/g ...