参考: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. Codeforces 559A(计算几何)

    A. Gerald's Hexagon time limit per test 2 seconds memory limit per test 256 megabytes input standard ...

  2. .net core webapi jwt 更为清爽的认证 ,续期很简单(2)

    .net core webapi jwt 更为清爽的认证  后续:续期以及设置Token过期 续期: 续期的操作是在中间件中进行的,续期本身包括了前一个Token的过期加发放新的Token,所以在说续 ...

  3. 一致性哈希算法-----> 解决memecache 服务器扩容后的数据丢失。

    1 基本场景 比如你有 N 个 cache 服务器(后面简称 cache ),那么如何将一个对象 object 映射到 N 个 cache 上呢,你很可能会采用类似下面的通用方法计算 object 的 ...

  4. P1003 铺地毯(noip 2011)

    洛谷——P1003 铺地毯 题目描述 为了准备一个独特的颁奖典礼,组织者在会场的一片矩形区域(可看做是平面直角坐标系的第一象限)铺上一些矩形地毯.一共有 n 张地毯,编号从 1 到n .现在将这些地毯 ...

  5. Google C++ style guide——C++类

    1.构造函数的职责 构造函数中仅仅进行那些没有实际意义的初始化.由于成员变量的"有意义"的值大多不在构造函数中确定. 能够的话,使用Init()方法集中初始化为有意义的数据. 长处 ...

  6. [Analytics] Add Tealium debugger in Chrome

    It would be helpful once you can see what information have been tracking inside you web application, ...

  7. 一个Exchange 2010 的password不定期弹框的问题处理,希望对大家可以有所帮助。

    前期有个客户採用exchange 2010的邮件系统作为他们的邮件平台.有个奇怪的现象就是Exchange 2010 系统会出现不定期"宕机"现象,为何打上引號,就是这个时候仅仅有 ...

  8. Ubuntu使用adb连接android手机失败unknown的解决的方法

    Ubuntu使用adb连接android手机失败unknown的解决的方法   Ubuntu下通过USB数据线连接G11手机后,adb可能无法识别到设备.依照一下步骤能够解决此问题. 1.在termi ...

  9. Visual Studio 2017中使用正则修改部分内容 如何使用ILAsm与ILDasm修改.Net exe(dll)文件 C#学习-图解教程(1):格式化数字字符串 小程序开发之图片转Base64(C#、.Net) jquery遍历table为每一个单元格取值及赋值 。net加密解密相关方法 .net关于坐标之间一些简单操作

    Visual Studio 2017中使用正则修改部分内容   最近在项目中想实现一个小工具,需要根据类的属性<summary>的内容加上相应的[Description]特性,需要实现的效 ...

  10. 【转】TestNG中的并发(多线程)

    优势 并行(多线程)技术在软件术语里被定义为软件.操作系统或者程序可以并行地执行另外一段程序中多个部分或者子组件的能力 多线程方式拥有很大的优势: 1). 减少测试运行时间 如果测试集里包含了大量的用 ...