「CF1188E」Problem from Red Panda

题目

点这里看题目。


给定一个长度为 \(k\) 的非负整数序列 \(a\)

你可以对于 \(a\) 做如下操作任意次:

  • 选定 \(1\le j\le k\),满足除了 \(a_j\)\(a\) 中其它数都为正

    而后,令 \(a_j\) 加上 \(k-1\),令除了 \(a_j\)\(a\) 中其它数减去 \(-1\)

    (这样一次操作被称为“操作 \(j\)”)

你需要求出经过任意次操作后,可能的 \(a\) 序列有多少种。答案对 \(998244353\) 取模。

所有数据满足 \(1\le k\le 10^5,0\le a_i\le 10^6\)

分析

一上来竟然想到 DAG 游走上面去了

注意到,最终得到的序列(不妨记为 \(a'\)和操作顺序无关,只和每个位置操作了多少次有关。具体来说,设 \(x_i\) 表示“操作 \(i\)”总共执行的次数,那么有 \(a'_i=a_i+kx_i-\sum_t x_t\)。在此,我们记 \(m=\sum_t x_t\),则 \(a'_i=a_i+kx_i-m\)

考虑 \(x\) 序列即可帮助我们避免算重的问题。注意到,对于 \(k\) 个数全部操作一遍等于没有操作,所以值得考虑的 \(x\) 必须满足 \(\min x=0\)。冷静一下,发现仅统计 \(\min x=0\) 的序列也就不会算重了。具体证明如下:

Proof.

考虑两个表示操作次数序列 \(x\)\(y\),满足 \(\min x=\min y=0\)\(x\neq y\)。我们同时记 \(m_x=\sum_t x_t,m_y=\sum_t y_t\)

如果 \(m_x=m_y\),因为 \(x\neq y\),所以容易推出它们产生的结果序列不同。

如果 \(m_x\neq m_y\),不妨设 \(m_x<m_y\)。反证,假设对于任意的下标 \(j\),都有 \(a_j+kx_j-m_x=a_j+ky_j-m_y\),于是 \(x_j<y_j\)于是 \(y_j>x_j\ge 0\),所以 \(\min y>0\),矛盾。

现在我们已经解决了一半的问题,考虑另一半——即“是否存在一个可行的操作序列”的问题。

做一些数值上的观察。注意到 \(a_i+kx_i-m\ge 0\Leftrightarrow x_i\ge \lceil\frac{\max\{m-a_i,0\}}{k}\rceil\)。进一步地,应该有 \(m=\sum_i x_i\ge \sum_i\lceil\frac{\max\{m-a_i,0\}}{k}\rceil\)

这个条件看起来还是不太强,因为它甚至弱于“\(a\) 中不能有两个及以上的 \(0\)”。但是,我们注意到进行任意多次操作后,这个条件都必须成立。也即,对于任意的 \(0\le t\le m\)\(t\ge \sum_i\lceil\frac{\max\{t-a_i,0\}}{k}\rceil\) 都应成立

这个条件够强吗?至少它不弱于“至多一个 \(0\)”。我们考虑建立归纳证明:

Conclusion.

对于操作次数序列 \(x\),记 \(m=\sum_i x_i\),则存在一种合法且“操作 \(i\)”出现次数为 \(x_i\) 的操作序列当且仅当 \(\forall 0\le t\le m,t\ge \sum_i\lceil\frac{\max\{t-a_i,0\}}{k}\rceil\)

Proof.

对于 \(m\) 进行归纳。首先,\(m=0\) 意味着 \(x\) 全为 \(0\),显然是成立的。

\(m>0\) 时,我们需要证明充分性和必要性:

必要性

我们只需要证明,存在一个 \(i_0\) 满足 \(x_{i_0}>0\) 且“操作 \(i_0\)”合法且操作后右侧条件仍然满足,即可生成一个操作序列。假设 \(i_0\) 存在,我们设操作后的序列为 \(b\)

\(w_t=\sum_{i}\lceil\frac{\max\{t-a_i,0\}}{k}\rceil\)。任取 \(0\le t<m\),有:

\[\begin{aligned} \sum_i\left\lceil\frac{\max\{t-b_i,0\}}{k}\right\rceil&=\sum_{i\neq i_0}\left\lceil\frac{\max\{t+1-a_i,0\}}{k}\right\rceil+\left\lceil\frac{\max\{t-a_{i_0}-k+1,0\}}{k}\right\rceil\\ &=w_{t+1}-\left\lceil\frac{\max\{t+1-a_{i_0},0\}}{k}\right\rceil+\left\lceil\frac{\max\{t-a_{i_0}-k+1,0\}}{k}\right\rceil \end{aligned} \]

如果 \(t+1-a_{i_0}>0\),那么就有 \(LHS=w_{t+1}-1\le t\);如果 \(t+1-a_{i_0}\le 0\),则 \(LHS=w_{t+1}\)

贪心地想,我们令 \(i_0\)使得 \(x_i>0\)\(a_i\) 最小的 \(i\)。注意到,如果 \(x_i=0\),则 根据 \(x_i\ge \lceil\frac{\max\{m-a_i,0\}}{k}\rceil\) 可知 \(a_i\ge m\)。于是,对于 \(t\in [0,a_{i_0})\) 而言,\(w_{t+1}\) 的和式中无论是 \(x_i>0\) 的项还是 \(x_i=0\) 的项贡献都必然为 \(0\),于是 \(w_{t+1}=0\)

那么,执行“操作 \(i_0\)”合法吗?不合法的情况是,除了 \(a_{i_0}\) 以外还有数为 \(0\)。考虑 \(w_1\ge 1\) 即可知此情况不存在。

故必要性得证。

充分性

此处,我们沿用“必要性”证明的 \(i_0\)

在此,我们不加证明地指出:

如果存在一种符合条件的操作序列,则必然存在一种操作序列,使得其第一次操作为“操作 \(i_0\)”。

类似于“必要性”证明的讨论,可以证明 \(t\in [0,m)\)\(w_{t+1}\le t+1\)。再讨论一下 \(t=0\) 的情况,可知 \(w_{0}=0\)

故充分性得证。

证明完这个重要结论之后,我们只需要枚举 \(s\) 并计算方案数即可。\(s\) 确定时,方案数可以直接插板得出。而当 \(s>\max a\) 时,对于任意 \(x_i\) 都有 \(x_i>0\),故 \(s\) 只需枚举到 \(\max a\)

复杂度因此为 \(O(\max a+k\log k)\)

代码

#include <bits/stdc++.h>

#define rep( i, a, b ) for( int i = (a) ; i <= (b) ; i ++ )
#define per( i, a, b ) for( int i = (a) ; i >= (b) ; i -- )

const int mod = 998244353;
const int MAXN = 2e6 + 5;

template<typename _T>
inline void Read( _T &x ) {
    x = 0; char s = getchar(); bool f = false;
    while( s < '0' || '9' < s ) { f = s == '-', s = getchar(); }
    while( '0' <= s && s <= '9' ) { x = ( x << 3 ) + ( x << 1 ) + ( s - '0' ), s = getchar(); }
    if( f ) x = -x;
}

template<typename _T>
inline void Write( _T x ) {
    if( x < 0 ) putchar( '-' ), x = -x;
    if( 9 < x ) Write( x / 10 );
    putchar( x % 10 + '0' );
}

int fac[MAXN], ifac[MAXN];

int su[MAXN];
int A[MAXN];

int K, V;

inline int Qkpow( int, int );
inline int Inv( const int &a ) { return Qkpow( a, mod - 2 ); }
inline int Mul( int x, const int &v ) { return 1ll * x * v % mod; }
inline int Sub( int x, const int &v ) { return ( x -= v ) < 0 ? x + mod : x; }
inline int Add( int x, const int &v ) { return ( x += v ) >= mod ? x - mod : x; }

inline int& MulEq( int &x, const int &v ) { return x = 1ll * x * v % mod; }
inline int& SubEq( int &x, const int &v ) { return ( x -= v ) < 0 ? ( x += mod ) : x; }
inline int& AddEq( int &x, const int &v ) { return ( x += v ) >= mod ? ( x -= mod ) : x; }

inline int C( int n, int m ) { return n < m ? 0 : Mul( fac[n], Mul( ifac[m], ifac[n - m] ) ); }

inline int Qkpow( int base, int indx ) {
    int ret = 1;
    while( indx ) {
        if( indx & 1 ) MulEq( ret, base );
        MulEq( base, base ), indx >>= 1;
    }
    return ret;
}

inline void Down( int &x ) { x &= x - 1; }
inline void Up( int &x ) { x += x & ( - x ); }
inline void Update( int x, int v ) { for( ; x <= K ; Up( x ) ) su[x] += v; }
inline  int Query( int x ) { int ret = 0; for( ; x ; Down( x ) ) ret += su[x]; return ret; }

inline void Init( const int &n ) {
    fac[0] = 1; rep( i, 1, n ) fac[i] = Mul( fac[i - 1], i );
    ifac[n] = Inv( fac[n] ); per( i, n - 1, 0 ) ifac[i] = Mul( ifac[i + 1], i + 1 );
}

int main() {
    Read( K );
    rep( i, 1, K )
        Read( A[i] ), V = std :: max( V, A[i] );
    std :: sort( A + 1, A + 1 + K );

    Init( V + K );
    int ans = 0, sub = 0;
    for( int s = 0, i = 1, j = 1 ; s <= V ; s ++ ) {
        for( ; i <= K && A[i] <= s ; i ++ )
            sub += A[i] / K, Update( A[i] % K + 1, +1 );
        for( ; j <= K && A[j] <  s ; j ++ );
        int low = s / K * ( i - 1 ) - sub + Query( s % K );
        if( low > s ) break;
        AddEq( ans, Sub( C( s - low + K - 1, K - 1 ), C( s - low + j - 2, K - 1 ) ) );
    }
    Write( ans ), putchar( '\n' );
    return 0;
}
posted @ 2023-04-26 21:03  crashed  阅读(38)  评论(0编辑  收藏  举报