HNOI2009有趣的数列
首先next_permutation打表,发现Cat规律。
其实考试的时候这么做没什么问题,而且可以节省异常多的时间,那么现在我们来想一下why。
首先我拿模型法解释一下,我们把2n个数看成2n个人,既然分成奇数和偶数两种比较方式,那么我让他们站成两排,每一排有n个人,这n个人的身高递增,且,第二排的人必须高于第一排,那么这个问题就变成了:
有2n个身高互不相同人站成两排,每排n人,要求右边的人比左边的人高,后面的人比前面的人高,问我有几种排队方案。
这是一个Cat的模型,既然先站哪一排无所谓,我就让某个位置必须先站上第一排的人再站上第二排的人,如果我将站在第一排看做是0,站在第二排看做是1,那么既然每个1前面一定有一个比他矮的人,则一定有一个0,那么就又转化成了求0,1序列,这是一个更加经典的Cat模型。(如果这里理解不了可以上网搜搜)
然后再拿折线法解释一下,我们把偶数项看做x轴上的数,因为他们是单增的,把奇数项看做y轴上的数,由于奇数项小于与之对应偶数项,也就是不能越过y=x,函数的变化就好像只能向右走和向上走。这个问题在上一篇博客中有详细的解法。
所以我们明白它是让我们求Cat,可是P不一定是质数,逆元的问题很恶心。
所以我们采用唯一分解来做。首先线性筛筛出2n以内的所有素数,然后我们枚举每个素数,对n执行以下操作:将n不断的除以这个素数,并将商加入s变量,最终s的值就是n!在算术基本定理拆分后,这个素数的指数。举个例子:
8!=27*32*5*7,8/2=4,4/2=2,2/2=1,1/2=0。4+2+1+0=7。
20!=218……,20/2=10,10/2=5,5/2=2,2/2=1,1/2=0。10+5+2+1+0=18。
大家可以自己随便试两个。
这是为什么呢?(下述i为质数)首先1~n中含有i这个因子的数有n/i个(1),含有i2这个因子的数有n/i2个(2),……含有im这个因子的数有n/im个(m)。那么我们分层计算贡献,首先(1)中有n/i个i,加上,(2)中有2*n/i2个i,但不要忘了,我们在(1)算过每个数中的一个i,那么它们的贡献只有n/i2个i,同理,向后类推,最后n!中i的个数为∑n/pi,与上述模拟过程一致。
那么我们来证明一下复杂度,首先根据小于N的质数约有N/lnN个,我们第一层枚举的代价就是O(N/lnN),然后观察上述过程,我们的问题规模不断缩小,如上述二例,都是1/2、1/2的速度在缩小,对于其他素数类似,我们取最坏O(log2N),那么总复杂度
O(N/lnN*log2N),这玩意换换底就是O(N/ln2),1/ln2≈1.44,撇掉,大约O(N),(这是我自己证的,网上目测没有,如果有异议请指出,应该没什么问题吧……)
然后分子加分母减拆完了拿快速幂一乘就完事了。(快速幂并不影响上述复杂度,因为qpow也是O(logk)的,就当常数大了吧)。
(底下代码有表机,勾掉的调试略多,可以用来自己见证一下上面那个算法的正确性)
#include<iostream>
#include<algorithm>
#include<cmath>
#include<cstring>
#include<cstdio>
#include<vector>
#include<queue>
#include<stack>
#include<set>
#include<map>
using namespace std;
int n,P,/*a[20],*/ans=;
/*bool check(){
for(int i=1;i<=n;i++)
if(a[i*2-1]>a[i*2]) return 0;
for(int i=3;i<=n*2;i++)
if(a[i]<a[i-2]) return 0;
return 1;
}*/
int prime[],prime_num;
bool v[];
void doprime(){
for(int i=;i<=*n+;i++){
if(!v[i]) prime[++prime_num]=i;
for(int j=;j<=prime_num&&i*prime[j]<=*n+;j++){
v[prime[j]*i]=;
if(i%prime[j]==) break;
}
}
}
int qpow(int x,int k){
int val=;
for(;k;k>>=,x=1ll*x*x%P)
if(k&) val=1ll*val*x%P;
return val%P;
}
int main(){
//打表找规律系列。。。
/* while(1){
ans=0;
scanf("%d%d",&n,&P);
for(int i=1;i<=(n<<1);i++)
a[i]=i;
do{
if(check()) {ans++;
for(int i=1;i<=2*n;i++)
cout<<a[i]<<" ";
cout<<endl;
}
}while(next_permutation(a+1,a+1+2*n));
printf("ANS=%d\n",ans);
}*/
scanf("%d%d",&n,&P);
doprime();
for(int i=;i<=prime_num;i++){
long long s=;
for(int j=*n;j/=prime[i];) s+=j;
// cout<<"s1="<<s<<endl;
for(int j=n;j/=prime[i];) s-=j;
//cout<<"s2="<<s<<endl;
for(int j=n+;j/=prime[i];) s-=j;
// cout<<"s3="<<s<<endl;
ans=1ll*ans*qpow(prime[i],s)%P;
}
// cout<<"Okprime"<<endl;
/* for(int i=1;i<=prime_num;i++)
cout<<prime[i]<<" ";cout<<endl;*/
/* for(int i=1;i<=2*n;i++)
mulfz(i);
for(int i=1;i<=n;i++)
mulfm(i);
for(int i=1;i<=n+1;i++)
mulfm(i);*/
/* cout<<"OKfenjie"<<endl;
for(int i=1;i<=prime_num;i++)
ans=1ll*ans*qpow(prime[i],fz[i]-fm[i])%P;
cout<<"Okqpow"<<endl;*/
printf("%d",ans);
return ;
}
这道题取模,下道题高精。
HNOI2009有趣的数列的更多相关文章
- BZOJ 1485: [HNOI2009]有趣的数列( catalan数 )
打个表找一下规律可以发现...就是卡特兰数...卡特兰数可以用组合数计算.对于这道题,ans(n) = C(n, 2n) / (n+1) , 分解质因数去算就可以了... -------------- ...
- BZOJ 1485: [HNOI2009]有趣的数列 [Catalan数 质因子分解]
1485: [HNOI2009]有趣的数列 Description 我们称一个长度为2n的数列是有趣的,当且仅当该数列满足以下三个条件: (1)它是从1到2n共2n个整数的一个排列{ai}: (2)所 ...
- BZOJ_1485_[HNOI2009]有趣的数列_卡特兰数
BZOJ_1485_[HNOI2009]有趣的数列_卡特兰数 Description 我们称一个长度为2n的数列是有趣的,当且仅当该数列满足以下三个条件: (1)它是从1到2n共2n个整数的一个排列{ ...
- 【BZOJ1485】[HNOI2009]有趣的数列(组合数学)
[BZOJ1485][HNOI2009]有趣的数列(组合数学) 题面 BZOJ 洛谷 题解 从小往大填数,要么填在最小的奇数位置,要么填在最小的偶数位置. 偶数位置填的数的个数不能超过奇数位置填的数的 ...
- [HNOI2009]有趣的数列 题解(卡特兰数)
[HNOI2009]有趣的数列 Description 我们称一个长度为2n的数列是有趣的,当且仅当该数列满足以下三个条件: (1)它是从1到2n共2n个整数的一个排列{ai}: (2)所有的奇数项满 ...
- [HNOI2009]有趣的数列 卡特兰数
题面:[HNOI2009]有趣的数列 题解: 观察到题目其实就是要求从长为2n的序列中选n个放在集合a,剩下的放在集合b,使得集合a和集合b中可以一一对应的使a中的元素小于b. 2种想法(实质上是一样 ...
- 「BZOJ1485」[HNOI2009] 有趣的数列 (卡特兰数列)
「BZOJ1485」[HNOI2009] 有趣的数列 Description 我们称一个长度为2n的数列是有趣的,当且仅当该数列满足以下三个条件: (1)它是从1到2n共2n个整数的一个排列{ai ...
- 洛谷P3200 [HNOI2009]有趣的数列(Catalan数)
P3200 [HNOI2009]有趣的数列 题目描述 我们称一个长度为2n的数列是有趣的,当且仅当该数列满足以下三个条件: (1)它是从1到2n共2n个整数的一个排列{ai}: (2)所有的奇数项满足 ...
- bzoj1485: [HNOI2009]有趣的数列(Catalan数)
1485: [HNOI2009]有趣的数列 Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 2105 Solved: 1117[Submit][Stat ...
- [HNOI2009]有趣的数列(卡塔兰数,线性筛)
[HNOI2009]有趣的数列 题目描述 我们称一个长度为2n的数列是有趣的,当且仅当该数列满足以下三个条件: (1)它是从1到2n共2n个整数的一个排列{ai}: (2)所有的奇数项满足a1< ...
随机推荐
- js时间格式化和相互转换
1. Thu Mar 07 2019 12:00:00 GMT+0800 (中国标准时间) 转换为 2019-03-07 12:00:00 const d = new Date(Thu Mar 07 ...
- SAP云平台上的ABAP编程环境里如何消费第三方服务
在ABAP On-Premises环境下,使用ABAP编程消费第三方服务,相信很多ABAP顾问都已经非常熟悉了,无非就是使用CL_HTTP_CLIENT或者CL_REST_HTTP_CLIENT来发送 ...
- 使用abapGit在ABAP On-Premises系统和SAP云平台ABAP环境之间进行代码传输
SAP ABAP顾问朋友们,应该都使用过SAPLink这个工具.如果两个ABAP Netweaver系统没有建立起传输路径时,我们无法使用标准的SE10事务码创建传输请求的方式进行这两个系统间的代码传 ...
- PS批量制作获奖证书并导出PNG
其实方法和"使用PS批量制作视频字幕"的方法类似.区别在于制作视频字幕时导出成psd格式就可以直接导入Premiere中使用了,而获奖证书考虑到打印设备有无PS的不确定性,可能需要 ...
- PHP危险函数的持续学习
记录下遇到过的PHP危险函数 0x01 escapeshellarg()与escapeshellsmd()联合 先给出官方的定义: escapeshellarg ( string $arg ) : s ...
- Java关键字—final
final—不可变的,用来说明最终属性,表明一个类不能派生出子类,或者成员方法不能被覆盖,或者成员域的值不能被改变. 修饰范围: 1.修饰类,被final关键字修饰的类无法被继承: 2.修饰方法,被f ...
- java容器一:Collection概述
Collection概览 java容器有两类,第一类是Collection,存储的是对象的集合:第二类是Map,存储的是键值对(两个对象以及它们之间的对应关系)的集合 Collection接口下面有三 ...
- C# 中自定义配置
微软在ConfigurationManager类里面为我们提供了AppSetting和ConnectionStrings 两个常用配置, 但是有时候我们需要自定的配置,例如 <image lef ...
- 用BCB 画 Code128 B模式条码
//--------------------------------------------------------------------------- #include <vcl.h> ...
- VGridControl 使用技巧
1. 让列的宽度自动填充 如果VGridControl的LayoutStyle属性为BandsView或SingleRecordView,那么把VGridControl的OptionsView.Aut ...