首页 > 其他分享 >快速求组合数的方法

快速求组合数的方法

时间:2022-11-22 21:09:44浏览次数:40  
标签:return 组合 int 快速 ll 逆元 fac 方法 mod


求C(n,m)%mod的方法总结

1.当n,m都很小的时候可以利用杨辉三角直接求。

C(n,m)=C(n-1,m)+C(n-1,m-1);

2.利用乘法逆元。

乘法逆元:(a/b)%mod=a*(b^(mod-2)) mod为素数。

逆元可以利用扩展欧几里德或欧拉函数求得:

1).扩展欧几里德:b*x+p*y=1 有解,x就是所求

2).费马小定理:b^(p-1)=1(mod p),故b*b^(p-2)=1(mod p),所以x=b^(p-2)

1. n!/(m!*(n-m)! =x%p ,先对算出n!、m!、(n-m)!对p取模的余数,就转换为a/b=x%p;因为p为素数,所以等价于bx+py=a;然后用扩展的欧几里得定理算出 bx’+py’=1的解,x=x’*a,就得到了最终的x的值,即C(m,n)%p得值。

2.逆元 其实如果mod是素数 则b的逆元其实就是b^(mod-2)

也就是 (m!(n-m)!)的逆元为 (m!(n-m)!)p-2 ;


int inv(int a)
{
//return fpow(a, MOD-2, MOD);
return a==1?1:(long long)(MOD-MOD/a)*inv(MOD%a)%MOD;
}
LL C(LL n,LL m)
{
if(m<0)return 0;
if(n<m)return 0;
if(m>n-m)m=n-m;
LL up=1,down=1;
for(LL i=0;i<m;i++)
{
up=up*(n-i)%MOD;
down=down*(i+1)%MOD;
}
return up*inv(down)%MOD;
}

3.当n和m比较大,mod是素数且比较小的时候(10^5左右),通过Lucas定理计算

Lucas定理:A、B是非负整数,p是质数。A B写成p进制:A=a[n]a[n-1]…a[0],B=b[n]b[n-1]…b[0]。 

则组合数C(A,B)与C(a[n],b[n])C(a[n-1],b[n-1])…*C(a[0],b[0]) mod p同余

即:Lucas(n,m,p)=C(n%p,m%p)*Lucas(n/p,m/p,p)


#include<iostream>
//#include<algorithm>
using namespace std;
typedef long long ll;
int quick_power_mod(int a,int b,int m)//pow(a,b)%m
{
int result = 1;
int base = a;
while(b>0)
{
if(b&1==1)
result=(result*base)%m;
base=(base*base)%m;
b>>=1;
}
return result;
}
//计算组合数取模
ll comp(ll a,ll b,int p)//composite num C(a,b)%p
{
if(a<b)return 0;
if(a==b)return 1;
if(b>a-b)b=a-b;
int ans=1,ca=1,cb=1;
for(ll i=0;i<b;i++)
{
ca=(ca*(a-i))%p;
cb=(cb*(b-i))%p;
}
ans=(ca*quick_power_mod(cb,p-2,p))%p;
return ans;
}
ll lucas(ll n,ll m,ll p)
{
ll ans=1;
while(n&&m&&ans)
{
ans=(ans*comp(n%p,m%p,p))%p;//also can be recusive
n/=p;
m/=p;
}
return ans;
}
int main()
{
ll m,n;
while(cin>>n>>m)
cout<<lucas(n,m,10007)<<endl;
return 0;
}

C(n % mod, m % mod) % mod; 如果太大还是利用上面的逆元来处理。

半预处理

由于Lucas定理保证了阶乘的数均小于p,所以可以讲所有的阶乘先预处理,优化C(n,m)

mod的要求:p<10^6,且为素数

有效范围:1<=n,m<=10^9


//半预处理 
const ll MAXN=100000;
ll fac[MAXN+100];
void init(int mod)
{
fac[0]=1;
for(int i=1;i<mod;i++)
fac[i]=fac[i-1]*i%mod;
}
//半预处理逆元求C(n,m)%mod
ll C(ll n,ll m)
{
if (m>n)return 0;
return fac[n]*(GetInverse(fac[m]*fac[n-m],mod))%mod;
}

可以预处理出阶乘和阶乘的逆元。

int C(int n,int m)
{
return 1ll*fac[n]*inv[m]%p*inv[n-m]%p;
}
fac[0]=inv[0]=inv[1]=1;
for(int i=1;i<=N;i++)fac[i]=1ll*fac[i-1]*i%p;
for(int i=2;i<=N;i++)inv[i]=1ll*(p-p/i)*inv[p%i]%p;
for(int i=1;i<=N;i++)inv[i]=1ll*inv[i-1]*inv[i]%p;

标签:return,组合,int,快速,ll,逆元,fac,方法,mod
From: https://blog.51cto.com/u_15888102/5878461

相关文章

  • JS前期数组、字符串、时间、定时器、DOM\BOM事件方法等总结
    1.字符串方法        .charAt(对应字符元素下标)——根据下标查找字符串内元素        .charCodeAt(对应字符元素下标)——根据下标查找字符串某元素在u......
  • error:could not open ...jvm.cfg解决方法
     出现这种情况大多是因为电脑上之前安装过JDK,卸载重装之后,运行java命令会出现error:couldnotopen…jvm.cfg的错误。打开系统环境变量,查看PATH,会看到诸如此类的配置......
  • 快速幂(板子)
    先讨论无需取模的当b为偶数时:ab=a(b/2)*2=(a2)b/2当b为奇数时:ab=a*ab-1=a*(a2)(b-1)/2如 28=(22)4     27=2*(22)3所以,我们可以如此迭代下......
  • jQuery-AJAX get() 和 post()方法,jQuery ajax() 方法
    什么是jQuery?jQuery是一个JavaScript函数库。jQuery是一个轻量级的"写的少,做的多"的JavaScript库。jQuery库包含以下功能:HTML元素选取HTML元素操作CSS......
  • 常用方法
    常用方法iterator.hasNext()、iterator.next()、iterator.remove()IteratorJSON.toJSONString(List<Student>);List转jso......
  • es组合查询
    POSTyi_cloud_ecs_monitor/_search{"query":{"bool":{"must":{"terms":{"storage_rate":["0","0"]}......
  • 外贸找客户三大必备"套路",让您快速开发客户
    开发外贸客户,成交是一件很艰难的事情,但同时也是一件很有成就感的事情,很多人觉得做外贸很难,那是因为找不到技巧。外贸成单重要的就是跟进客户,我们要通过不断的跟进客户才能获......
  • TSMaster快速入门篇(2)-报文回放
    支持格式TSMaster的数据回放默认支持blf格式(未来会增加对其他格式的支持)。如果需要分析其他数据格式的log文件,需要通过文件转换器从其他格式转成blf格式。一、离线回......
  • [译]Golang中JSON和结构体的组合使用
     原文地址:http://attilaolah.eu/2014/09/10/json-and-struct-composition-in-go/ 假设你正在把一个JSON对象解码为Go的结构体。该JSON来自不受你控制的服......
  • 前端性能优化方法
    什么是前端性能优化(what)?从用户访问资源到资源完整的展现在用户面前的过程中,通过技术手段和优化策略,缩短每个步骤的处理时间从而提升整个资源的访问和呈现速度。###为什么......