ABC 132 | D - Blue and Red Balls

题目描述

给定K个蓝球和NK个红球,这N个球从左到右排成一列,Takahashi可以每次选连续的若干个蓝球直至蓝球被选完,他会在尽可能少的步数内完成。
问有多少种球的摆放方式使得Takahashi可以在恰好i(1iK)步内取完所有蓝球?由于方案数很多,输出mod1e9+7后的结果。

数据范围

1KN2000

题解

本题是一个计数题,可以通过排列组合直接进行计算。

  • 首先将K个蓝球分为i组,有Ck1i1种方案数
  • 然后考虑用红球将蓝球间隔开,有以下三种方案。
    • NK个红球分为i组,有CNK1i1种方案,考虑到i组红球与i组蓝球交替排放有两种方案,故总方案数为2CNK1i1
    • NK个红球分为i1组,有CNK1i2种方案
    • NK个红球分为i+1组,有CNK1i种方案
  • 综上,最后的方案数有Ck1i1(2CNK1i1+CNK1i2+CNK1i)种。

但若直接按照上述公式写成代码,在避免RE错误加特判的过程中会导致答案错误,当然这是由于对计算过程理解不够导致的,这种情况可以先化简公式,也可以构造特殊样例发现错误

Ck1i1(2CNK1i1+CNK1i2+CNK1i)=Ck1i1(CNK1i1+CNK1i2+CNK1i1+CNK1i))=Ck1i1CNKi1+CNKi)=Ck1i1CNK+1i

使用的化简公式是Cnm=Cn1m1+Cn1m

代码

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>

using namespace std;

typedef long long ll;

const int N = 2010, mod = 1e9 + 7;

int n, k;
ll c[N][N];

void init(){
    for(int i = 0; i < N; i ++)
        for(int j = 0; j <= i; j ++)
            if(!j) c[i][j] = 1;
            else c[i][j] = (c[i - 1][j] + c[i - 1][j - 1]) % mod;
}

int main()
{
    init();
    scanf("%d%d", &n, &k);
    for(int i = 1; i <= k; i ++){
        ll ans = c[k - 1][i - 1] * c[n - k + 1][i] % mod;
        printf("%lld\n", ans);
    }
    return 0;
}

posted @   小菜珠的成长之路  阅读(36)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义
· 地球OL攻略 —— 某应届生求职总结
· 提示词工程——AI应用必不可少的技术
· Open-Sora 2.0 重磅开源!
· 周边上新:园子的第一款马克杯温暖上架
点击右上角即可分享
微信分享提示