ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

cf1622 D. Shuffle

2022-05-16 01:00:08  阅读:170  来源: 互联网

标签:ones 01 Shuffle cout 5000 cf1622 signed continue


题意:

给定01串,你可以进行至多一次操作:选择一个恰含 K 个1的子段,任意改变这个子段中字符的顺序。问可能得到的不同串的数量

串长 5000

思路:

正常计数很容易就重复计算了。正解是暴力:

枚举第一个被改变(0变1、1变0)的位置 \(l\) 和最后一个被改变的位置 \(r\)(\(l<r\))。注意只要 \([l,r]\) 中 1 的数量 \(\le K\) 就行。

\(l,r\) 一定要变,然后 \([l,r]\) 中还剩 \(ones\) 个 1 待分配,要把他们分配到 \((l,r)\) 上,也就是 \(C_{r-l-1}^{ones}\)

const signed N = 3 + 5000, mod = 998244353;
ll n, K, C[N][N], a[N], s[N]; //01串前缀和
signed main() {
    iofast;
    cin >> n >> K;

    for(int i = 0; i <= n; i++) { //initC
        C[i][0] = 1;
        for(int j = 1; j <= i; j++)
            C[i][j] = (C[i-1][j-1] + C[i-1][j]) % mod;
    }

    for(int i = 1; i <= n; i++) {
        char c; cin >> c;
        a[i] = c-'0', s[i] = s[i-1] + a[i];
    }

    if(s[n] < K) return cout << 1, 0; //特判1太少的情况

    ll ans = 1;
    for(int l = 1; l <= n; l++)
        for(int r = l + 1; r <= n; r++) {
            int ones = s[r] - s[l-1];
            if(ones > K) continue;
            ones -= (a[l] == 0) + (a[r] == 0);
            if(ones < 0) continue;
            ans += C[r-l-1][ones];
        }
    cout << ans % mod;
}

标签:ones,01,Shuffle,cout,5000,cf1622,signed,continue
来源: https://www.cnblogs.com/wushansinger/p/16275447.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有