• Allowed Letters CodeForces


    题意:

      给出一个字符串 给出几个定点必须是哪个字母(或者是几个字母中的一个)  然后求在满足所有定点后的最小字符串

    解析:

      没错 这题是暴力 用状压暴力 

       “a - f” 用”0 - 5“ 这几个数字代替   输入字符串  num[i]为字母i的个数,然后输入定点必须为哪个字母,ti[i]中用六位二进制来存储当前位置i的可以放的字母

    1代表放 0不放

      设cnt = 1<<6 - 1; 即为所有状态的上界 

      然后从后向前遍历一遍,统计对于位置i到n  状态k还需要多少个 sum[i][j]来表示

      然后暴力就好了。。就是从前向后遍历每一个位置,对于位置i 我们遍历所有字母挨个试,又对于每一个字母,统计一下在这个位置放了这个字母后,我们去检查是否还能满足后边的所有状态(挨个状态遍历一遍求出当前状态所需要的字母总和判断一下) 如果可以 则下一个位置  不可以下一个试字母 如果所有字母都不符合 那就impossible

    #include <bits/stdc++.h>
    #define mem(a, b) memset(a, b, sizeof(a))
    using namespace std;
    const int maxn = 100100, INF = 0x7fffffff, cnt = (1<<6) - 1;
    char str[maxn];
    int num[6], ti[maxn*10], sum[maxn][cnt+10], zt[cnt+10];
    int m;
    vector<char> v;
    
    int main()
    {
        cin>> str;
        int n, len;
        n = len = strlen(str);
        for(int i=0; i<len; i++)
            num[str[i]-'a']++;  //统计每个字母的数量
        cin>> m;
        int id;
        for(int i=0; i<m; i++)
        {
            cin>> id >> str;
            len = strlen(str);
            for(int j=0; j<len; j++)
            {
                ti[id-1] |= (1<<(str[j] - 'a'));    //统计能放在位置id-1 的字母集合
            }
        }
        for(int i=0; i<n; i++)
            if(!ti[i]) ti[i] = cnt;     //如果为0 说明没有被指定 所以都可以放
        for(int i=n-1; i>=0; i--)       //从后向前遍历一遍,统计对于位置i到n  状态k还需要多少个
        {
            for(int j=1; j<=cnt; j++)   //对当前i 遍历所有状态
            {
                if((ti[i] & j) == ti[i])    //子集
                    zt[j]++;
                sum[i][j] = zt[j];
            }
        }
        int _sum = 0;
        for(int i=0; i<n; i++)     //遍历每个位置 暴力判断即可
        {
            int flag1 = 0;
            for(int j=0; j<6; j++)
            {
                int flag2 = 1;
                if(!num[j] || (ti[i]&(1<<j)) == 0) continue;
                num[j]--;
                for(int k=1; k<=cnt; k++)
                {
                    _sum = 0;
                    for(int r=0; r<6; r++)
                        if(k & (1<<r))
                            _sum += num[r];
                    if(_sum < sum[i+1][k])
                    {
                        flag2 = 0; break;
                    }
                }
                if(flag2)
                {
                    flag1 = 1; v.push_back('a'+j); break;
                }
                num[j]++;
            }
            if(!flag1)
                return puts("Impossible"), 0;
        }
        for(int i=0; i<v.size(); i++)
        {
            printf("%c", v[i]);
        }
        cout<<endl;
    
        return 0;
    }
    自己选择的路,跪着也要走完。朋友们,虽然这个世界日益浮躁起来,只要能够为了当时纯粹的梦想和感动坚持努力下去,不管其它人怎么样,我们也能够保持自己的本色走下去。
  • 相关阅读:
    java基础5 (一维)数组和二维数组
    Java 内部类
    Java 抽象类和接口
    Java 多态
    Java 继承
    Java 包(package)
    Java String类和StringBuffer类
    Java 数组
    Java 封装与类
    Java 概述和编程基础
  • 原文地址:https://www.cnblogs.com/WTSRUVF/p/9637008.html
Copyright © 2020-2023  润新知