2018.09.25 bzoj1856: [Scoi2010]字符串(组合数学)

传送门
如果有n==m的条件就是卡特兰数。
但现在n不一定等于m。
我们可以考虑用求卡特兰数一样的方法来求答案。
我们知道有一种求卡特兰数的方法是转到二维平面求答案。
这道题就可以这样做。
我们将这个序列映射到二维平面上。
相当于从 ( 0 , 0 ) (0,0) (0,0)出发,每次只能向右上方或者向右下方走对应着选1/0,最后应该停在 ( n + m , n − m ) (n+m,n-m) (n+m,nm)
但这样会出现非法状态。
如何排除?
我们发现如果出现非法状态一定会穿过直线 y = − 1 y=-1 y=1,这样我们把图像关于 y = − 1 y=-1 y=1翻折,起点就变成了 ( 0 , − 2 ) (0,-2) (0,2)
这样总方案数是 ( n + m n ) \binom {n+m} {n} (nn+m),不合法的就是 ( n + m m − 1 ) \binom {n+m} {m-1} (m1n+m)
代码:

#include<bits/stdc++.h>
#define mod 20100403
#define ll long long
using namespace std;
inline ll ksm(ll x,int p=mod-2,ll ret=1){
	while(p){
		if(p&1)ret=ret*x%mod;
		x=x*x%mod,p>>=1;
	}
	return ret;
}
ll n,m;
inline ll C(ll x,ll y){
	ll ret=1,tmp=1;
	for(int i=1;i<=x;++i)ret=ret*i%mod;
	for(int i=1;i<=y;++i)tmp=tmp*i%mod;
	(ret*=ksm(tmp))%=mod,tmp=1;
	for(int i=1;i<=x-y;++i)tmp=tmp*i%mod;
	return (ret*ksm(tmp))%mod;
}
int main(){
	scanf("%lld%lld",&n,&m);
	printf("%lld",(C(n+m,n)-C(n+m,m-1)+mod)%mod);
	return 0;
} 
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值