zoukankan      html  css  js  c++  java
  • Numbers Exchange

    题意:

    Eugeny有n张卡片,他希望和Nikolay交换一些卡片使得他拥有的奇数数字和偶数数字的卡片数目一样,且所有数字都不同。

    Nikolay有m张卡片,分别写着1到m。问最少交换几次,能够满足要求。并输出交换后的结果。无解输出-1,多解任意输出一组。

    解法:

    贪心,首先用没有出现过的数字填好重复出现的数字,并顺便尽量让odd,even差值较小。

    然后贪心用没出现的数字填即可。

    #include <iostream>
    #include <cstdio>
    #include <cstring>
    #include <set>
    
    #define N 200010
    
    using namespace std;
    
    int n, m, a[N], j = 2, k = 1;
    set<int> mp, Tp;
    
    int get(int x)
    {
        if(x == 0)
        {
            while(j <= m && mp.count(j)) j += 2;
            if(j <= m) return j;
            else return 0;
        }
        else
        {
            while(k <= m && mp.count(k)) k += 2;
            if(k <= m) return k;
            else return 0;
        }
    }
    
    int main()
    {
        scanf("%d %d", &n, &m);
        int cnt0 = 0, cnt1 = 0, ans = 0;
        for(int i = 1;i <= n;i++)
        {
            scanf("%d", &a[i]);
            if(a[i] % 2 == 0) cnt0++;
            else cnt1++;
            mp.insert(a[i]);
        }
    //    cout << cnt0 << ' ' << cnt1 << endl;
        for(int i = 1;i <= n;i++)
        {
            if(Tp.count(a[i]))
            {
                if(a[i]&1) cnt1--;
                else cnt0--;
                int tmp0 = get(0), tmp1 = get(1);
                if(cnt0 < cnt1 && tmp0) a[i] = tmp0;
                else if(cnt1 < cnt0 && tmp1) a[i] = tmp1;
                else if(tmp0) a[i] = tmp0;
                else if(tmp1) a[i] = tmp1;
                else
                {
                    puts("-1");
                    return 0;
                }
                ans++;
                mp.insert(a[i]);
                if(a[i]&1) cnt1++;
                else cnt0++;
            }
            Tp.insert(a[i]);
        }
    //    cout << "ans = " << ans << endl;
    //    for(int i = 1;i <= n;i++) cout << a[i] << ' ';
    //    cout << endl << cnt0 << ' ' << cnt1 << endl;
        for(int i = 1;i <= n && cnt0 != cnt1;i++)
        {
            if(cnt0 < cnt1 && a[i] % 2 == 1)
            {
                cnt0++;
                cnt1--;
                int tmp = get(0);
                if(tmp) a[i] = tmp, ans++;
                else
                {
                    puts("-1");
                    return 0;
                }
            }
            if(cnt1 < cnt0 && a[i] % 2 == 0)
            {
                cnt0--;
                cnt1++;
                int tmp = get(1);
                if(tmp) a[i] = tmp, ans++;
                else
                {
                    puts("-1");
                    return 0;
                }
            }
            mp.insert(a[i]);
        }
        cout << ans << endl;
        for(int i = 1;i <= n;i++) cout << a[i] << ' ';
        cout << endl;
        return 0;
    }
    View Code
  • 相关阅读:
    Floyd_Warshall算法
    Bellman_Ford算法
    深度优先搜索
    广度优先搜索
    贪心算法_活动选择
    动态规划_0-1背包问题
    算法导论_动态规划_最长回文子序列
    算法导论_动态规划_最长公共子序列
    动态规划解决分割问题
    2016 Google中国开发者大会游记
  • 原文地址:https://www.cnblogs.com/lawyer/p/6782960.html
Copyright © 2011-2022 走看看