【Henu ACM Round#18 B】Modulo Sum

【链接】 我是链接,点我呀:)
【题意】

在这里输入题意

【题解】

m比较小
<=1000

a[i]直接看成a[i]%m就可以了。

有n个0..999之间的整数。。

如果有一个0那么就直接输出Yes.

否则要用那些数字凑0

则用cnt[0..999]记录每个数字出现的个数。
即n个物品,每个物品cnt[i]个。
然后凑和为0

->多重背包。

但每个物品的数量可能很多。
所以加一个二进制优化就好了。
把每个物品的数量转化成二进制。

转换成01背包的问题。
(物品的数目大概在1000*log2(1e6)的样子

然后容量是1000.

这时候就可以做了。

另解
如果n>m的话。
前缀和数组中肯定有sum[i]%m和sum[j]%m相同。
(抽屉原理,每个数字在0..m-1之间,然后有m+1个数字的话,显然不可能每个数字都相同。
则i+1..j之间的和就是m的倍数了。
所以直接输出YES.
否则做一个O(M^2)的背包就好。

【代码】

#include <bits/stdc++.h>
#define ll long long
using namespace std;

const int M = 1e3+10;

bool f[2][M];
int a[M],n,m;
vector<int> v;

int main()
{
    ios::sync_with_stdio(0),cin.tie(0);
    #ifdef LOCAL_DEFINE
        freopen("rush.txt","r",stdin);
    #endif
    cin >> n >> m;
    for (int i = 1;i <= n;i++) {
            int x;cin >>x;x%=m;
            a[x]++;
    }

    if (a[0]>0){
        cout<<"YES"<<endl;
        return 0;
    }

    for (int i = 1;i < m;i++)
        if (a[i]>0){
            int temp = 1;
            while (a[i]>=temp){
                v.push_back((1LL*temp*i)%m);
                a[i]-=temp;
                temp*=2;
            }
            if (a[i]>0) v.push_back((1LL*a[i]*i)%m);
        }
    if (!v.empty()){
        f[0][v[0]] = 1;
        if (v[0]==0) return cout<<"YES"<<endl,0;
        for (int i = 1;i <(int)v.size();i++){
            for (int j = 0;j < m;j++) f[i&1][j] = 0;
            for (int j = 1;j <= m-1;j++)
                if (f[(i&1)^1][j]){
                    int k = (j+v[i])%m;
                    f[i&1][j] = 1;
                    f[i&1][k] = 1;
                }
            f[i&1][v[i]] = 1;
            if (f[i&1][0]) return cout<<"YES"<<endl,0;
        }
    }
    cout<<"NO"<<endl;
    return 0;
}

原文地址:https://www.cnblogs.com/AWCXV/p/8376551.html

时间: 2024-08-27 04:06:32

【Henu ACM Round#18 B】Modulo Sum的相关文章

【Henu ACM Round #13 C】 Ebony and Ivory

[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 先求出c-bx的所有可能 ->存在map里面 然后枚举y看看ay在不在map里面 在的话就有解. 这样复杂度是\(O(N*log_2N)\)的 比直接两层循环枚举的\(O(N^2)\)复杂度要来的好 这种方法也叫"中途相遇法" [代码] #include <bits/stdc++.h> #define ll long long using namespace std; ll a,b,c; map<

【Henu ACM Round #13 B】Spider Man

[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 会发现每个环可以操作的次数是固定的. (环的大小-1 也就是说一旦环确定了.其实结果就已经确定了. 则直接看操作总数的奇偶性就可以了. [代码] #include <bits/stdc++.h> #define long long using namespace std; int n,x; int main(){ #ifdef LOCAL_DEFINE freopen("rush_in.txt", &quo

【Henu ACM Round #13 D】A Trivial Problem

[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 123...n中末尾0的个数 只会由素因子2和5的个数决定且等于 Min{cnt[2],cnt[5]} 且素因子2的个数一定会比5多; 所以n!的末尾0的个数 等于for (int i = 1;i <= n;i++)中所有i的5因子的个数和 枚举一下就好 [代码] #include <bits/stdc++.h> using namespace std; const int N = 1e7; int n; vector&

【Henu ACM Round #13 E】Spy Syndrome 2

[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 对m个串建立一棵字典树. 然后对主串. 尝试从第一个位置开始.在字典树中尝试匹配 如果匹配到了位置i 就再从位置i+1开始尝试匹配 (这时又重新从根节点开始重新匹配 每次匹配最多只要往下走50步. 写个递归的过程就好. [代码] #include <bits/stdc++.h> using namespace std; const int N = 1e6; const int NN = 1e4; const int M = 1

【Henu ACM Round#14 A】Vitaly and Night

[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 连续两个如果不全是0就递增cnt [代码] #include <bits/stdc++.h> using namespace std; int n,m; int main() { cin >> n >>m; int cnt = 0; for (int i = 1;i <= n;i++){ int x,y; for (int j = 1;j <= m;j++){ cin >>x &

【Henu ACM Round#14 C】Duff and Weight Lifting

[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 2^y可以由两个2^(y-1)相加得到. 则有一个贪心的策略. 就是2^x尽量都变成2^(x+1) (即能够凑就尽量凑) 如果x还有剩余的话.答案递增1 而凑上去的数字,显然是可以合并成1步操作的.因为他们的和就是2^(x+1) [代码] #include <bits/stdc++.h> #define ll long long using namespace std; const int N = 1e6; int n,a[N

【Henu ACM Round#15 E】 A and B and Lecture Rooms

[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 最近公共祖先. (树上倍增 一开始统计出每个子树的节点个数_size[i] 如果x和y相同. 那么直接输出n. 否则求出x和y的最近公共祖先.z (假定y的深度大于x [1]如果z等于x或y中的一个. 那么久就找到x..y的路径(长度设为L)中的中点u. 显然,u和它的其他len-1个子树上的任意一个节点都是可行的(除了那个包含y的子树 设_get(x,step)表示x节点往上走step步到达的节点 则输出_sum[中点]-_s

【Henu ACM Round#15 A】 A and B and Chess

[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 统计大写和小写的个数. 比较答案.输出即可. [代码] #include <bits/stdc++.h> using namespace std; string s[10]; map<char,int> dic; int inc[300]; int main() { for (int i = 0;i < 8;i++) cin >> s[i]; for (int i = 0;i < 8;i++

【Henu ACM Round#15 C】 A and B and Team Training

[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 枚举第一种方法. 剩下的全都个第二种方法. 看看能组成多少个队伍就可以了. [代码] #include <bits/stdc++.h> #define ll long long using namespace std; int n,m; int main() { ios::sync_with_stdio(0),cin.tie(0); cin >> n >> m; int ans = 0; for(int