Luogu 1641[SCOI2010]生成字符串 - 卡特兰数

Description

有$N$ 个 $1$ 和 $M$ 个 $0$ 组成的字符串, 满足前 $k$ 个字符中 $1$ 的个数不少于 $0$ 的个数。

求这样字符串的个数。

$1<=M <=N<=1e6$

Solution

正难则反, 很难直接求出满足条件的字符串的个数, 就从反面考虑。

$N$个$1$ 和 $M$ 个 $0$ 组成的字符串总共有 $C(N + M, N)$ 个, 再减去不满足条件的 字符串的个数就能够得到答案了。

不满足条件的字符串个数为$C(N+M,N+1)$

证明与 卡特兰数的证明类似:

设一个 不满足条件的字符串 $0$ 的个数 比 $1$ 多 的 位置为 $k$。

并且 对于任意 $j <k$, 前$j$个字符中$num_1>=num_0$, 而前$k$个字符$num_1<num_0$。

很显然 $num_0=num_1+1$, 我们将这个 $k$ 个字符都取反, $0$ 变成 $1$, $1$ 变成 $0$。

$0$ 的个数减少 $1$ 个, $1$ 的个数 增加 $1$个,

那么取反后的字符串 中 $1$ 的个数为 $N+1$, $0$ 的个数为 $M-1$。

由于 $N+1$个$1$ ,$M-1$个$0$ 组成的字符串恰好有$C(N+M,N+1)$个。

所以我们接下来要证明 它们是 一 一对应的,(即$(N,M)$中不满足条件的字符串通过转换能 对应上 $(N+1, M-1)$ 每种字符串)

  现在已知 $(N,M)$ 能通过转换变成 $(N+1,M-1)$ 且没有重复。

  只需证明$(N+1,M-1)$ 通过转换 变成 $(N,M)$ : 找到第一个 $1$ 比 $0$ 多的位置 $k$ 并把前 $k$ 个字符取反 即可。

证毕

最后答案就是 $C(N+M,N)-C(N+M,N+1)$。

Code

 1 #include<cstdio>
 2 #include<cstring>
 3 #include<algorithm>
 4 #define ll long long
 5 using namespace std;
 6
 7 const int mod = 20100403;
 8 const int N = 2e6 + 5;
 9
10 ll fpow(ll a, ll b) {
11     ll re = 1;
12     for (; b; b >>= 1, a = a * a % mod)
13         if (b & 1) re = re * a % mod;
14     return re;
15 }
16
17 ll fac[N], ans;
18
19 int main()
20 {
21     int n, m, M;
22     scanf("%d%d", &n, &m);
23     M = n + m;
24     fac[0] = fac[1] = 1;
25     for (int i = 1; i <= M; ++i)
26         fac[i] = fac[i - 1] * i % mod;
27     ans = fac[M] * fpow(fac[n], mod - 2) % mod;
28     ans = ans * fpow(fac[m], mod - 2) % mod;
29     ll tmp = fac[M] * fpow(fac[n + 1], mod - 2) % mod;
30     tmp = tmp * fpow(fac[m - 1], mod - 2) % mod;
31     ans = ans - tmp;
32     ans = (ans % mod + mod) % mod;
33     printf("%lld\n", ans);
34 }

  

原文地址:https://www.cnblogs.com/cychester/p/9718007.html

时间: 2024-11-05 16:07:01

Luogu 1641[SCOI2010]生成字符串 - 卡特兰数的相关文章

[SCOI2010]生成字符串 题解(卡特兰数的扩展)

[SCOI2010]生成字符串 Description lxhgww最近接到了一个生成字符串的任务,任务需要他把n个1和m个0组成字符串,但是任务还要求在组成的字符串中,在任意的前k个字符中,1的个数不能少于0的个数.现在lxhgww想要知道满足要求的字符串共有多少个,聪明的程序员们,你们能帮助他吗? 输入格式:输入数据是一行,包括2个数字n和m; 输出格式:输出数据是一行,包括1个数字,表示满足要求的字符串数目,这个数可能会很大,只需输出这个数除以20100403的余数; Solution 1

1856: [Scoi2010]字符串 卡特兰数

1856: [Scoi2010]字符串 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 1458  Solved: 814[Submit][Status][Discuss] Description lxhgww最近接到了一个生成字符串的任务,任务需要他把n个1和m个0组成字符串,但是任务还要求在组成的字符串中,在任意的前k个字符中,1的个数不能少于0的个数.现在lxhgww想要知道满足要求的字符串共有多少个,聪明的程序员们,你们能帮助他吗? Input

BZOJ1856 [SCOI2010]生成字符串 【组合数】

题目 lxhgww最近接到了一个生成字符串的任务,任务需要他把n个1和m个0组成字符串,但是任务还要求在组成的字符串中,在任意的前k个字符中,1的个数不能少于0的个数.现在lxhgww想要知道满足要求的字符串共有多少个,聪明的程序员们,你们能帮助他吗? 输入格式 输入数据是一行,包括2个数字n和m 输出格式 输出数据是一行,包括1个数字,表示满足要求的字符串数目,这个数可能会很大,只需输出这个数除以20100403的余数 输入样例 2 2 输出样例 2 提示 [数据范围] 对于30%的数据,保证

【洛谷P1641】[SCOI2010]生成字符串

题目描述 lxhgww最近接到了一个生成字符串的任务,任务需要他把n个1和m个0组成字符串,但是任务还要求在组成的字符串中,在任意的前k个字符中,1的个数不能少于0的个数.现在lxhgww想要知道满足要求的字符串共有多少个,聪明的程序员们,你们能帮助他吗? 输入输出格式 输入格式: 输入数据是一行,包括2个数字n和m 输出格式: 输出数据是一行,包括1个数字,表示满足要求的字符串数目,这个数可能会很大,只需输出这个数除以20100403的余数 输入输出样例 输入样例#1: 2 2 输出样例#1:

[SCOI2010]生成字符串

OJ题号: BZOJ1856.洛谷1641 思路: 总方案数为$\binom{n+m}{m}$,非法方案数为$\binom{n+m}{m-1}$. 则合法方案数为$(n-m+1)\frac{(n+2)(n+3)...(n+m)}{m!}$. 其中除以$m!$可以用乘以逆元实现,边乘边模. 因为要求出1~m的连续的逆元,所以可以线性推. 另外注意中间结果和最终乘积可能会超过int范围,所以要转long long,不然只有10分. 1 #include<cstdio> 2 #include<

NPU 2015年陕西省程序设计竞赛网络预赛(正式赛)F题 和谐的比赛(递推 ||卡特兰数(转化成01字符串))

Description 今天西工大举办了一场比赛总共有m+n人,但是有m人比较懒没带电脑,另外的n个人带了电脑.不幸的是,今天机房的电脑全坏了只能用带的电脑,一台电脑最多两人公用,确保n>=m.但是大家来的时间不同,随机次序来机房,带电脑的人直接准备比赛而没带电脑的人需要向带电脑并还没和别人公用的人求助(当然会答应).但是,如果不存在带电脑并还没和别人公用的人,那他就要等了,等是很让人头疼的,这就不和谐了,当然假如没有这样的情况发生比赛是很和谐的. Input 输入多组数据,每组数据只有一行m(

洛谷 p1044 栈 【Catalan(卡特兰数)】【经典题】

题目链接:https://www.luogu.org/problemnew/show/P1044 转载于:https://www.luogu.org/blog/QiXingZhi/solution-p1044 题目背景 栈是计算机中经典的数据结构,简单的说,栈就是限制在一端进行插入删除操作的线性表. 栈有两种最重要的操作,即pop(从栈顶弹出一个元素)和push(将一个元素进栈). 栈的重要性不言自明,任何一门数据结构的课程都会介绍栈.宁宁同学在复习栈的基本概念时,想到了一个书上没有讲过的问题,

hdu 5177 (1e18范围的卡特兰数)

hdu 5177 (1e18范围的卡特兰数) 题意: 求第n个卡特兰数,模3814697265625 (5^18) 限制: 有20组数据,1 <= n <= 1e18 思路: 1. 卡特兰数的表达式: ans = 1/(n+1) * C(2*n,n) -> ans = 1/(n+1) * (2n)! / n! / n!    ---1式 2. 因为要模5^18,求逆元要求互质,所以先把"1式"中的因子5全部去掉 3. 然后看不含因子5的阶乘,f(n!) 4. 设g(x

HDU 1023 Train Problem II (卡特兰数,经典)

题意:给出一个数字n,假设火车从1~n的顺序分别进站,求有多少种出站序列. 思路:卡特兰数的经典例子.n<101,用递推式解决.需要使用到大数.n=100时大概有200位以下. 1 #include <bits/stdc++.h> 2 using namespace std; 3 const int N=101; 4 vector<string> vect; 5 void _mult(string num1, string num2, string &result )