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

题目描述

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,n?m),这相当于从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,n?m),这个方案数可以用组合数表示为C(n+m,m-1)C(n+m,m?1)。

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

代码

我是用费马小定理做的

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
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;
}
    
时间: 2024-12-22 08:01:31

【洛谷P1641】[SCOI2010]生成字符串的相关文章

洛谷P1132 数字生成计划 广搜

洛谷P1132 数字生成计划 广搜 三种操作 因为要步数最少,所以广搜 1 #include <bits/stdc++.h> 2 #define For(i,j,k) for(int i=j;i<=k;i++) 3 using namespace std ; 4 5 const int N = 1000011 ; 6 struct node{ 7 int a,ans ; 8 }; 9 bool flag[N] ; 10 int f[N] ; 11 queue<node> q;

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

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

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

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

并查集 洛谷P1640 [SCOI2010]连续攻击游戏

[SCOI2010]连续攻击游戏 题目描述 lxhgww最近迷上了一款游戏,在游戏里,他拥有很多的装备,每种装备都有2个属性,这些属性的值用[1,10000]之间的数表示.当他使用某种装备时,他只能使用该装备的某一个属性.并且每种装备最多只能使用一次.游戏进行到最后,lxhgww遇到了终极boss,这个终极boss很奇怪,攻击他的装备所使用的属性值必须从1开始连续递增地攻击,才能对boss产生伤害.也就是说一开始的时候,lxhgww只能使用某个属性值为1的装备攻击boss,然后只能使用某个属性值

洛谷P2569 [SCOI2010]股票交易

P2569 [SCOI2010]股票交易 题目描述 最近lxhgww又迷上了投资股票,通过一段时间的观察和学习,他总结出了股票行情的一些规律. 通过一段时间的观察,lxhgww预测到了未来T天内某只股票的走势,第i天的股票买入价为每股APi,第i天的股票卖出价为每股BPi(数据保证对于每个i,都有APi>=BPi),但是每天不能无限制地交易,于是股票交易所规定第i天的一次买入至多只能购买ASi股,一次卖出至多只能卖出BSi股. 另外,股票交易所还制定了两个规定.为了避免大家疯狂交易,股票交易所规

洛谷 P3709 大爷的字符串题

https://www.luogu.org/problem/show?pid=3709 题目背景 在那遥远的西南有一所学校 /*被和谐部分*/ 然后去参加该省省选虐场 然后某蒟蒻不会做,所以也出了一个字符串题: 题目描述 给你一个字符串a,每次询问一段区间的贡献 贡献定义: 每次从这个区间中随机拿出一个字符x,然后把x从这个区间中删除,你要维护一个集合S 如果S为空,你rp减1 如果S中有一个元素不小于x,则你rp减1,清空S 之后将x插入S 由于你是大爷,平时做过的题考试都会考到,所以每次询问

洛谷P1132 数字生成游戏

P1132 数字生成游戏 题目描述 小明完成了这样一个数字生成游戏,对于一个不包含0的数字s来说,有以下3种生成新的数的规则: 将s的任意两位对换生成新的数字,例如143可以生成314,413,134: 将s的任意一位删除生成新的数字,例如143可以生成14,13,43 在s的相邻两位之间s[i],s[i + 1]之间插入一个数字x,x需要满足s[i] < x < s[i + 1].例如143可以生成1243,1343,但是不能生成1143,1543等. 现在小明想知道,在这个生成法则下,从s

洛谷——P1640 [SCOI2010]连续攻击游戏

P1640 [SCOI2010]连续攻击游戏 题目描述 lxhgww最近迷上了一款游戏,在游戏里,他拥有很多的装备,每种装备都有2个属性,这些属性的值用[1,10000]之间的数表示.当他使用某种装备时,他只能使用该装备的某一个属性.并且每种装备最多只能使用一次.游戏进行到最后,lxhgww遇到了终极boss,这个终极boss很奇怪,攻击他的装备所使用的属性值必须从1开始连续递增地攻击,才能对boss产生伤害.也就是说一开始的时候,lxhgww只能使用某个属性值为1的装备攻击boss,然后只能使