• CF思维训练,2020长春CCPC(A,D)


    D. Knowledge Cards(华容道GAME)

    题意:给定n*m的矩阵,初始在左上角有一摞牌,从上到下每张的数字构成了一个长度为k的排列,请问在保证1.不能使得牌在移动过程中重叠2.不能让牌反复出现在左上和右下角。这两个条件下,能否将所有的卡牌通过移动来使得最终摞在右下角的卡牌是一个自上而下的1,2,3,,,k的顺序

    思路:结论题,不难看出只要目前场上只要有一个空我就可以通过一系列的移动使得某张牌到我想要的位置上,所以只要目前要去右下角的牌放上去之后还能有一个空就可以了。

    #include 
    
    using namespace std;
    
    const int N = 1e6+100;
    int n,m,k;
    int a[N];
    int tr[N];
    int ask(int x) {
        int res = 0;
        for(x; x ; x -= (x & -x)) res += tr[x];
        return res;
    }
    
    void add(int x, int v) {
        for(x; x <= k ; x += (x & -x)) tr[x] += v;
    }
    
    void solve()
    {
        cin >> n >> m >> k;
        for(int i = 1; i <= k ; i ++ ) cin >> a[i], tr[i] = 0;
        bool flag = true;
        for(int i = 1; i <= k ; i ++ ) {
            if(ask(a[i]) >= n * m - 3) flag = false;
            add(a[i], 1);
        }
        if(flag) cout << "YA" << endl;
        else cout << "TIDAK" << endl;
    }
    
    signed main()
    {
        int t;
        for(cin>>t;t;t--)
        {
            solve();
        }
        return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40

    C. Almost All Multiples(构造)

    题意:给定n,x,请你构造一个排列,使得a[n]=1并且a[1]=x,而且要求每个位置上的数都是这个位置的倍数(即a[i]%i==0)的情况下使得排列的字典序最小。

    思路:思路太多了反而给自己写挂了,反思一下应该想清楚后再写

    对于给定的这个x;
    1.如果n%x!=0,,显然无法构造,必然有个位置不满足a[i]%i== 0
    2. 考虑如何构造,对于必然的两个点1和n,我们先构造好他,然后把n安放在x处,随后不断地向后试探是否有位置可以和n交换,交换后再不断试探,如此往复即可。

    #include 
    
    using namespace std;
    #define endl '\n'
    const int N = 2e5+100;
    int a[N];
    int main()
    {
        cin.tie(0);cout.tie(0);ios::sync_with_stdio(0);
        int t;
        for(cin>>t;t;t--)
        {
            int n,x;
            cin>>n>>x;
            for(int i=1;i<=n;i++)
                a[i]=0;
            if(n%x)
            {
                cout<<-1<<endl;
                continue;
            }
            for(int i=1;i<=n;i++)
            {
                a[i]=i;
            }
            if(n==x)
            {
                swap(a[1],a[n]);
                for(int i=1;i<=n;i++)
                    cout<<a[i]<<" ";
                cout<<endl;
                continue;
            }
            a[1]=x;
            a[n]=1;
            a[x]=n;
            int now=x;
            for(int i=2;i<n;i++)
            {
                if(n%i==0&&i%now==0)
                {
                    swap(a[now],a[i]);
                    now=i;
                }
            }
            for(int i=1;i<=n;i++)
            {
                cout<<a[i]<<" ";
            }
            cout<<endl;
        }
        return 0;
    }
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54

    真的气,为什么vp每次都有牌,济南就犯了傻逼错误,我求求老天了,让我们杭州正式赛发挥好一次,呜呜呜。
    A. Krypton多重背包

    题意:给定了n个物品,这n个物品你可以无限卖,第一次卖的时候每个物品会有不同的赠品,每个物品和赠品都有其价值,请问给定m块钱怎样买能获得最大价值。

    思路:zsy读到这题好像在想别的方法,和我说一下一眼多重背包就过了。

    顺带一提,所谓基础的四个背包(01,多重,无限,混合)其实都可以直接用多重背包来做,没那么麻烦的。

    #include 
    
    using namespace std;
    #define endl '\n'
    
    const int N = 1e5+100;
    struct node
    {
        int w,v,num;
    };
    node a[N];
    int dp[N];
    int main()
    {
        cin.tie(0);cout.tie(0);ios::sync_with_stdio(0);
        int n;
        cin>>n;
        a[1].num=1,a[1].v=18,a[1].w=1;
        a[2].num=1,a[2].v=78,a[2].w=6;
        a[3].num=1,a[3].v=280+28,a[3].w=28;
        a[4].num=1,a[4].v=880+58,a[4].w=88;
        a[5].num=1,a[5].v=1980+128,a[5].w=198;
        a[6].num=1,a[6].v=3280+198,a[6].w=328;
        a[7].num=1,a[7].v=6480+388,a[7].w=648;
        a[8].num=10000,a[8].v=10,a[8].w=1;
        a[9].num=10000,a[9].v=60,a[9].w=6;
        a[10].num=10000,a[10].v=280,a[10].w=28;
        a[11].num=10000,a[11].v=880,a[11].w=88;
        a[12].num=10000,a[12].v=1980,a[12].w=198;
        a[13].num=10000,a[13].v=3280,a[13].w=328;
        a[14].num=10000,a[14].v=6480,a[14].w=648;
        for(int i=1;i<=14;i++)
        {
            int num=min(a[i].num,n/a[i].w);
            for(int k=1;num>0;k<<=1)
            {
                if(k>num)
                    k=num;
                num-=k;
                for(int j=n;j>=a[i].w*k;j--)
                {
                    dp[j]=max(dp[j],dp[j-a[i].w*k]+a[i].v*k);
                }
            }
        }
        cout<<dp[n];
        return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48

    D. Meaningless Sequence(数位DP的思想,位运算,打表)
    题意:请对于a的数列,有如下通项公式。给定n和c请你求解第二个公式

    请添加图片描述在这里插入图片描述
    思路:俗话说得好,看不出规律就打表。
    打表发现a的每一项可以理解为c的多少次幂,而这个多少次幂又恰好发现是对应n的二进制表示下1的数量之和。那么这题就变成了:统计从0到n各个数二进制位都出现了多少次,用到了一点数位DP的思想

    #include 
    
    using namespace std;
    #define endl '\n'
    #define int long long
    const int N=3e3+5;
    const int mod = 1e9+7;
    int x,a[N],c[N][N];
    char s[N];
    int fastpow(int n,int a)
    {
        int res=1;
        while(n)
        {
            if(n&1)
            {
                res=res*a%mod;
            }
            a=a*a%mod;
            n>>=1;
        }
        return res%mod;
    }
    signed main()
    {
        ios::sync_with_stdio(0);cin.tie(0);cout.tie(0);
        c[0][0]=c[1][1]=1;
        for(int i=1;i<=3000;i++)
        {
            c[0][i]=c[i][i]=1;
            for(int j=0;j<i;j++)
            {
                c[i][j]=(c[i-1][j]+c[i-1][j-1])%mod;
            }
        }
        cin>>(s+1)>>x;
        int n=strlen(s+1),sum=0;
        for(int i=1;i<=n;i++)
        {
            if(s[i]=='0') continue;
            int len=n-i;
            for(int j=0;j<=len;j++)
                a[j+sum]=(a[j+sum]+c[len][j])%mod;
            sum++;
        }
        a[sum]++;
        int ans=0;
        for(int i=0;i<=3000;i++)
        {
            ans=(ans+a[i]*fastpow(i,x)%mod)%mod;
            //cout<
        }
        cout<<ans;
        return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
  • 相关阅读:
    anaconda python 版本对应关系
    QGraphicsView使用要点
    2327. 知道秘密的人数;1722. 执行交换操作后的最小汉明距离;2537. 统计好子数组的数目
    【Python】文件操作
    面试 - 判断对象属性是否存在
    第一次写计算机论文无从下手怎么办?(二) - 易智编译EaseEditing
    图像质量指标:PSNR、SSIM、MSE
    基于多目标优化算法的 LCOE电力成本的敏感性分析(Matlab代码实现)
    BUU [HCTF 2018]Hideandseek
    大整数相加,相减,相乘,大整数与普通整数的相乘,相除
  • 原文地址:https://blog.csdn.net/qq_35866893/article/details/128122914