zoukankan      html  css  js  c++  java
  • Codeforces Round #612 (Div. 2)

    C. Garland

    题意

    给你了一个序列,包含n个数,这个序列是由1~n数字构成,但是题目给你的这个序列并不完整,让你去补完整,那些输入的值为0的位置的就是让你去填数字,然后问你怎么填,这个序列的奇偶值最小。(一个序列的奇偶值大小就是这个序列的奇数和偶数的遇见次数,例如1,4,2,3,5奇偶值就是2,因为只有1,4和2,3两处奇数和偶数相遇)。

    思路

    开了一个四维的dp,dp[i][j][k][2];
    第一维度表示当前的位数
    第二维度表示当前已经填了多少个奇数
    第三维度表示当前已经填了多少个偶数
    第四维度表示当前填的这个数是奇数还是偶数。
    dp维度表示好了状态转移方程就很清楚了,当奇数和偶数相遇时就加一。

    代码

    #include<stdio.h>
    #include<string.h>
    #include<algorithm>
    using namespace std;
    const int N=105;
    int dp[N][N][N][2];
    int pre[N];
    int a[N];
    int os,js;
    int main()
    {
        int n;
        scanf("%d",&n);
        os=n/2;
        js=n-os;
        for(int i=1; i<=n; i++)
        {
            scanf("%d",&a[i]);
            if(a[i]>0)
            {
                if(a[i]%2==0)
                {
                    os--;
                }
                else
                {
                    js--;
                }
                pre[i]=pre[i-1];
            }
            else
            {
                pre[i]=pre[i-1]+1;
            }
        }
        memset(dp,0x3f3f3f,sizeof(dp));
        dp[0][0][0][1]=0;
        dp[0][0][0][0]=0;
        for(int i=1; i<=n; i++)
        {
            for(int j=0; j<=js; j++)
            {
                for(int k=0; k<=os; k++)
                {
                    if(a[i]<=0)
                    {
                        if(j+k<=pre[i])
                        {
                            if(j==0&&k!=0)
                            {
                                dp[i][j][k][0]=min(dp[i-1][j][k-1][0],dp[i-1][j][k-1][1]+1);
                                dp[i][j][k][1]=min(dp[i-1][j][k][0]+1,dp[i-1][j][k][1]);
                            }
                            else if(k==0&&j!=0)
                            {
                                dp[i][j][k][0]=min(dp[i-1][j][k][0],dp[i-1][j][k][1]+1);
                            dp[i][j][k][1]=min(dp[i-1][j-1][k][0]+1,dp[i-1][j-1][k][1]);
                            }
                            else if(k==0&&j==0)
                            {
                                dp[i][j][k][0]=min(dp[i-1][j][k][0],dp[i-1][j][k][1]+1);
                            dp[i][j][k][1]=min(dp[i-1][j][k][0]+1,dp[i-1][j][k][1]);
                            }
                            else
                            {
                                dp[i][j][k][0]=min(dp[i-1][j][k-1][0],dp[i-1][j][k-1][1]+1);
                            dp[i][j][k][1]=min(dp[i-1][j-1][k][0]+1,dp[i-1][j-1][k][1]);
                            }
     
                        }
                    }
                    else
                    {
                        if(j+k<=pre[i])
                        {
                            if(a[i]%2==0)
                            {
                                dp[i][j][k][0]=min(dp[i-1][j][k][0],dp[i-1][j][k][1]+1);
                            }
                            else
                            {
                                dp[i][j][k][1]=min(dp[i-1][j][k][0]+1,dp[i-1][j][k][1]);
                            }
                        }
     
                    }
                }
            }
        }
        printf("%d
    ",min(dp[n][js][os][0],dp[n][js][os][1]));
        return 0;
    }
    
    

    B. Young Explorers

    题意
    从一些数据中选三组满足一定关系的字符串,暴力是三层循环,一般要转化为两层,然后得到第三个。

  • 相关阅读:
    查看文件的首尾行
    快捷键
    lua中奇葩用法
    lua中特殊用法
    CVPR2015一些文章整理
    hdu5371 最长回文子串变形(Manacher算法)
    SVN合并(merge)的使用
    atitit. 集合groupby 的实现(2)---自己定义linq查询--java .net php
    王立平--SQLite,SQLiteOpenHelper的简单应用
    Vim 经常使用快捷键及键盘图
  • 原文地址:https://www.cnblogs.com/wangqianyv/p/13279758.html
Copyright © 2011-2022 走看看