参考:http://blog.csdn.net/wzq_qwq/article/details/46709471

首先推组合数,设sum为每个人礼物数的和,那么答案为

\[( C_{n}^{sum}C_{sum}^{w[1]}c_{sum-w[1]}^{w[2]}...
\]

设w[0]=n-sum,然后化简成阶乘的形式:

\[\frac{n!}{w[0]!w[1]!...w[n]!}
\]

注意到这里p不是质数,所以把p拆成质数的方相乘的形式,最后用中国剩余定理合并即可

然后现在的问题是怎么快速求出阶乘

假设当前的质数的方为p=3那么1x2x3x4x5x6x7x8x9x10x11=1x2x4x5x7x8x10x11x 3x(1x2x3),注意到后面又是一个阶乘,但是范围更小,所以可以递归来做,然后前面乘的3被模消去了

#include<iostream>
#include<cstdio>
using namespace std;
const int N=100005;
long long P,n,m,w[10],p[N],cnt[N],mod[N],tot,sum,a[N];
struct qwe
{
int a,b;
};
void exgcd(long long a,long long b,long long &x,long long &y,long long &d)
{
if(!b)
{
x=1;
y=0;
d=a;
return;
}
exgcd(b,a%b,y,x,d);
y=y-a/b*x;
}
long long china()
{
long long d,x=0,y;
for(int i=1;i<=tot;i++)
{
long long r=P/mod[i];
exgcd(mod[i],r,d,y,d);
x=(x+r*y*a[i])%P;
}
return (x+P)%P;
}
long long ksm(long long a,long long b,long long mod)
{
long long r=1ll;
while(b)
{
if(b&1)
r=r*a%mod;
a=a*a%mod;
b>>=1;
}
return r;
}
long long inv(long long a,long long b)
{
long long x,y,d;
exgcd(a,b,x,y,d);
return (x%b+b)%b;
}
qwe fac(long long k,long long n)
{
qwe r;
if(!n)
{
r.a=0,r.b=1;
return r;
}
long long x=n/p[k],y=n/mod[k],ans=1ll;
if(y)
{
for(int i=2;i<mod[k];i++)
if(i%p[k]!=0)
ans=ans*i%mod[k];
ans=ksm(ans,y,mod[k]);
}
for(int i=y*mod[k]+1;i<=n;i++)
if(i%p[k]!=0)
ans=ans*i%mod[k];
qwe tmp=fac(k,x);
r.a=x+tmp.a,r.b=ans*tmp.b%P;
return r;
}
long long clc(int k,long long n,long long m)
{
if(n<m)
return 0;
qwe a=fac(k,n),b=fac(k,m),c=fac(k,n-m);
return ksm(p[k],a.a-b.a-c.a,mod[k])*a.b%mod[k]*inv(b.b,mod[k])%mod[k]*inv(c.b,mod[k])%mod[k];
}
long long wk(long long n,long long m)
{
for(int i=1;i<=tot;i++)
a[i]=clc(i,n,m);
return china();
}
int main()
{
scanf("%lld%lld%lld",&P,&n,&m);
for(int i=1;i<=m;i++)
scanf("%lld",&w[i]),sum+=w[i];
int x=P;
for(int i=2;i*i<=x;i++)
if(x%i==0)
{
p[++tot]=i;
mod[tot]=1;
while(x%i==0)
{
x/=i;
cnt[tot]++;
mod[tot]*=i;
}
}
if(x>1)
{
p[++tot]=x;
mod[tot]=x;
cnt[tot]=1;
}
if(sum>n)
{
puts("Impossible");
return 0;
}
long long ans=wk(n,sum)%P;
for(int i=1;i<=m;i++)
{
ans=ans*wk(sum,w[i])%P;
sum-=w[i];
}
printf("%lld\n",ans);
return 0;
}

bzoj 2142: 礼物【中国剩余定理+组合数学】的更多相关文章

  1. BZOJ 2142 礼物 组合数学 CRT 中国剩余定理

    2142: 礼物 Time Limit: 10 Sec  Memory Limit: 259 MBSubmit: 1450  Solved: 593[Submit][Status][Discuss] ...

  2. BZOJ 2142: 礼物 [Lucas定理]

    2142: 礼物 Time Limit: 10 Sec  Memory Limit: 259 MBSubmit: 1294  Solved: 534[Submit][Status][Discuss] ...

  3. BZOJ - 2142 礼物 (扩展Lucas定理)

    扩展Lucas定理模板题(貌似这玩意也只能出模板题了吧~~本菜鸡见识鄙薄,有待指正) 原理: https://blog.csdn.net/hqddm1253679098/article/details ...

  4. 【刷题】BZOJ 2142 礼物

    Description 一年一度的圣诞节快要来到了.每年的圣诞节小E都会收到许多礼物,当然他也会送出许多礼物.不同的人物在小E 心目中的重要性不同,在小E心中分量越重的人,收到的礼物会越多.小E从商店 ...

  5. bzoj 2142 礼物——扩展lucas模板

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2142 没给P的范围,但说 pi ^ ci<=1e5,一看就是扩展lucas. 学习材料 ...

  6. BZOJ 2142: 礼物

    模非素数下的排列组合,简直凶残 调着调着就过了= = 都不知道怎么过的= = 直接上链接http://hi.baidu.com/aekdycoin/blog/item/147620832b567eb4 ...

  7. BZOJ.2142.礼物(扩展Lucas)

    题目链接 答案就是C(n,m1) * C(n-m1,m2) * C(n-m1-m2,m3)...(mod p) 使用扩展Lucas求解. 一个很简单的优化就是把pi,pi^ki次方存下来,因为每次分解 ...

  8. BZOJ 2142 礼物 数论

    这道题是求组合数终极版. C(n,m) mod P n>=1e9 m>=1e9 P>=1e9且为合数且piqi<=1e5 拓展lucas定理. 实际上就是一点数论小知识的应用. ...

  9. BZOJ 1951 [SDOI2010]古代猪文 (组合数学+欧拉降幂+中国剩余定理)

    题目大意:求$G^{\sum_{m|n} C_{n}^{m}}\;mod\;999911659\;$的值$(n,g<=10^{9})$ 并没有想到欧拉定理.. 999911659是一个质数,所以 ...

随机推荐

  1. Xcode waring: no rule to process file *** 警告提示

    在编译程序的时候,Xcode给出了警告:warning: no rule to process file *** 类似的警告, 解决方法: 在[build Phases] -> [Compile ...

  2. 前端学习之--CSS

    CSS 常用帮助文档 CSS:被用来同时控制多重网页的样式和布局.HTML页面中CSS样式的写法有3种: 1:标签中写入 <body style='margin o auto;'> 2:h ...

  3. PAT (Advanced Level) 1038. Recover the Smallest Number (30)

    注意前导零的消去. #include <iostream> #include <string> #include <sstream> #include <al ...

  4. linux下查看哪个进程占用内存多

    1.用top命令 1.top top命令是Linux下常用的性能分析工具,能够实时显示系统中各个进程的资源占用状况,类似于Windows的任务管理器 可以直接使用top命令后,查看%MEM的内容.可以 ...

  5. POJ 3264 Balanced Lineup(RMQ_ST)

    题目链接:http://poj.org/problem? id=3264 Description For the daily milking, Farmer John's N cows (1 ≤ N  ...

  6. IOS开发 APP提交程序上传流程

    由于苹果的机制,在非越狱机器上安装应用必须通过官方的App Store,开发者开发好应用后上传App Store,也需要通过审核等环节.AppCan作为一个跨主流平台的一个开发平台,也对ipa包上传A ...

  7. webpack-Module Resolution(模块解析)

    模块解析(Module Resolution) resolver 是一个库(library),用于帮助找到模块的绝对路径.一个模块可以作为另一个模块的依赖模块,然后被后者引用,如下: import f ...

  8. hdu 5087 Revenge of LIS II ( LIS ,第二长子序列)

    链接:hdu 5087 题意:求第二大的最长升序子序列 分析:这里的第二大指的是,全部的递增子序列的长度(包含相等的), 从大到小排序后.排在第二的长度 cid=546" style=&qu ...

  9. Redis 命令行 常用总结

    http://www.redis.cn/commands.html# 1 Keys * 列出所有的keys redis > keys * ) "s:0" ) "o: ...

  10. unique函数(STL)

    unique()函数是一个去重函数,STL中unique的函数 unique的功能是去除相邻的重复元素(只保留一个),还有一个容易忽视的特性是它并不真正把重复的元素删除.他是c++中的函数,所以头文件 ...