博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
【洛谷P1641】[SCOI2010]生成字符串
阅读量:4538 次
发布时间:2019-06-08

本文共 1998 字,大约阅读时间需要 6 分钟。

题目描述

lxhgww最近接到了一个生成字符串的任务,任务需要他把n个1和m个0组成字符串,但是任务还要求在组成的字符串中,在任意的前k个字符中,1的个数不能少于0的个数。现在lxhgww想要知道满足要求的字符串共有多少个,聪明的程序员们,你们能帮助他吗?

输入输出格式

输入格式:

输入数据是一行,包括2个数字n和m

输出格式:

输出数据是一行,包括1个数字,表示满足要求的字符串数目,这个数可能会很大,只需输出这个数除以20100403的余数

输入输出样例

输入样例#1:
2 2
输出样例#1:
2

说明

limitation

每点2秒

对于30%的数据,保证1<=m<=n<=1000

对于100%的数据,保证1<=m<=n<=1000000

来源:SCOI 2010

分析(转载)

写的确实不错。本题要数形结合,有一定的技巧是一个不错的题。

首先,我们设选1为(1,1),选0为(1,-1)

目标就是(n+m,n-m)

总方案数为C(n+m,n),因为有n+m个位置,放n个1

然后要减去不合法的即线路通过y=-1的。将线路与y=-1交点的左边沿着y=-1做对称操作,则最后等价于从(0,-2)走到(n+m,n-m)的方案数

所以向上走n-m+2

则有x-y=n-m+2

  x+y=n+m

  x=n+1,y=m-1

所以不合法方案为C(n+m,n+1)

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

求这些用模逆元,O(n)求解

 

另附上洛谷题解中的分析

可以考虑把11的个数与00的个数的和看成xx坐标,11的个数与00的个数的差看成yy坐标,那么如下图:

向右上走(xx坐标加11,yy坐标加11)就表示这个字符选择11。

向右下走(xx坐标加11,yy坐标减11)就表示这个字符选择00。

这样子,如果不考虑限制条件,就表示从(0,0)(0,0)走n+mn+m步到达(n+m,n-m)(n+m,nm),这相当于从n+mn+m步中选出mm步向右下走,也就是C(n+m,m)C(n+m,m)。

考虑限制条件,任意前缀中11的个数不少于00的个数,也就是这条路径不能经过直线y=-1y=1。可以通过对称性发现,从(0,0)(0,0)走到直线y=-1y=1上的一点,相当于从(0,-2)(0,2)走到该点。也就是说,路径经过直线y=-1y=1的方案数就是从(0,-2)(0,2)走n+mn+m步到达(n+m,n-m)(n+m,nm),这个方案数可以用组合数表示为C(n+m,m-1)C(n+m,m1)。

所以最后结果为C(n+m,m)-C(n+m,m-1)C(n+m,m)C(n+m,m1)。对于组合数,可以预处理阶乘后用乘法逆元计算。

代码

我是用费马小定理做的

#include
#include
#include
#include
using namespace std;const int p=20100403;typedef long long ll;int n,m;int fac[2000005];int power(int a,int b){ int res=1,base=a; while(b) { if(b&1) res=(ll)res*(ll)base%(ll)p; base=(ll)base*(ll)base%(ll)p; b=b>>1; } return res;}int inv(int x){ return power(x,p-2);}void fct(){ fac[0]=1; for(int i=1;i<=2000000;i++) fac[i]=(ll)fac[i-1]*(ll)i%(ll)p;}int main(){ scanf("%d%d",&n,&m); fct(); int t1,t2,x1,x2,ans; t1=fac[n+m]; t2=(ll)fac[n]*(ll)fac[m]%(ll)p; x1=(ll)t1*(ll)inv(t2)%(ll)p; t2=(ll)fac[m-1]*(ll)fac[n+1]%(ll)p; x2=(ll)t1*(ll)inv(t2)%(ll)p; ans=(x1%p-x2%p+p)%p; printf("%d\n",ans); return 0;}

 

转载于:https://www.cnblogs.com/huihao/p/7449960.html

你可能感兴趣的文章
HDU5092——Seam Carving(动态规划+回溯)(2014上海邀请赛重现)
查看>>
语言基础
查看>>
C# : 操作Word文件的API - (将C# source中的xml注释转换成word文档)
查看>>
C#中字符串转换成枚举类型的方法
查看>>
Airplace平台
查看>>
TinyOS实例介绍
查看>>
我是怎么定义微服务平台?
查看>>
python random
查看>>
input输入框只允许输入数字/ 数字+小数点/ 文字+字母/ 等解决方法
查看>>
【翻译】西川善司「实验做出的游戏图形」「GUILTY GEAR Xrd -SIGN-」中实现的「纯卡通动画的实时3D图形」的秘密,前篇(2)...
查看>>
mysql 5.6 参数详解
查看>>
求旋转数组的最小元素
查看>>
Gson解析Json数组
查看>>
Lintcode: Fast Power
查看>>
Pocket Gem OA: Log Parser
查看>>
枚举也能直接转换为对应的数值输出
查看>>
angularjs1-7,供应商
查看>>
BitSet
查看>>
Spring常用注解,自动扫描装配Bean
查看>>
(转载)深入理解WeakHashmap
查看>>