题目描述

n 名学生排成了一列长队,每名学生都戴口罩,口罩的式样共有 m 种。每一名学生都比较关心排在他前面的学生中,有多少人带的口罩式样和他是相同的。设 ai
表示在第 i 名学生前的 i1 名学生中,有 ai 名学生戴的口罩式样与他一模一样,请问有多少种不同的排队方案,能够使得与给定数据一致?输出答案对 109+7 取模,若不可能出现输入的情况,则输出 0

限制:

  • 1n105
  • 1m100

算法分析

C[i] 表示恰染了 i 次的同种颜色的颜色数量

考察以下这个例子:

A=[0,0,1,0,1,1,2]

一开始没有染色,所以 C[0]=3
对于第一个口罩,可以染的颜色有 3 种,所以可以把 3 乘进答案里,此时 C[0]=2C[1]=1
对于第二个口罩,可以染的颜色有 2 种,所以可以把 2 乘进答案里,此时 C[0]=1C[1]=2
对于第三个口罩,可以染的颜色有 2 种,所以可以把 2 乘进答案里,此时 C[0]=1C[1]=1C[2]=1
对于第四个口罩,可以染的颜色有 1 种,所以可以把 1 乘进答案里,此时 C[0]=0C[1]=2C[2]=1
对于第五个口罩,可以染的颜色有 2 种,所以可以把 2 乘进答案里,此时 C[0]=0C[1]=1C[2]=2
对于第六个口罩,可以染的颜色有 1 种,所以可以把 1 乘进答案里,此时 C[0]=0C[1]=0C[2]=3
对于第七个口罩,可以染的颜色有 3 种,所以可以把 3 乘进答案里,此时 C[0]=0C[1]=0C[2]=2C[3]=1.

所以最后的答案就是 3×2×2×1×2×1×3=72

代码实现
#include <bits/stdc++.h>
#define rep(i, n) for (int i = 0; i < (n); ++i)
using namespace std;
using ll = long long;
//const int mod = 998244353;
const int mod = 1000000007;
struct mint {
ll x;
mint(ll x=0):x((x%mod+mod)%mod) {}
mint operator-() const {
return mint(-x);
}
mint& operator+=(const mint a) {
if ((x += a.x) >= mod) x -= mod;
return *this;
}
mint& operator-=(const mint a) {
if ((x += mod-a.x) >= mod) x -= mod;
return *this;
}
mint& operator*=(const mint a) {
(x *= a.x) %= mod;
return *this;
}
mint operator+(const mint a) const {
return mint(*this) += a;
}
mint operator-(const mint a) const {
return mint(*this) -= a;
}
mint operator*(const mint a) const {
return mint(*this) *= a;
}
mint pow(ll t) const {
if (!t) return 1;
mint a = pow(t>>1);
a *= a;
if (t&1) a *= *this;
return a;
}
// for prime mod
mint inv() const {
return pow(mod-2);
}
mint& operator/=(const mint a) {
return *this *= a.inv();
}
mint operator/(const mint a) const {
return mint(*this) /= a;
}
};
istream& operator>>(istream& is, mint& a) {
return is >> a.x;
}
ostream& operator<<(ostream& os, const mint& a) {
return os << a.x;
}
int main() {
int n, m;
cin >> n >> m;
vector<int> a(n);
rep(i, n) cin >> a[i];
vector<int> c(n);
c[0] = m;
mint ans = 1;
rep(i, n) {
if (!c[a[i]]) {
puts("0");
return 0;
}
ans *= c[a[i]];
c[a[i]]--;
c[a[i] + 1]++;
}
cout << ans << '\n';
return 0;
}