zoukankan      html  css  js  c++  java
  • 【Uva 10118】Free Candies

    Link:

    Description

    有4堆书;
    每本书编号从1..20
    每堆书都是N本;
    然后每次只能从任意一堆的堆顶拿一本书装到自己的口袋里;
    你的口袋最多容纳5本书;
    当你的口袋里有两本一样的书的时候,那一对书就归你了;
    但是一旦你的口袋装满了,就不能再装书了;游戏停止
    问你最多能拿多少对书。

    Solution

    比较明显的动规了;
    每次只能从4堆书的堆顶中选择一本书
    定义f[i][j][k][l]表示第一堆书上有i本数,第二堆书。。。。获得的最大书对数;
    则从某一堆书上拿一本书;
    相当于某一维变量的值减去1;
    写个前缀和,记录最初始每一堆书的第j本书上面1..20这些书各有多少本
    转移的时候,就能根据这个前缀和,获取之前拿了哪些书了;
    成对的消掉,增加数目,不能成对的,则记录书的类型;
    对于拿的书的类型<5的状态做转移就好;

    NumberOf WA

    0

    Reviw


    Code

    #include <bits/stdc++.h>
    using namespace std;
    
    const int N = 40;
    const int INF = 0x3f3f3f3f;
    
    int n,x[5][N+10],f[N+5][N+5][N+5][N+5],pre[5][N+10][21];
    int temp[22];
    
    pair<int,int> tongji(int a,int b,int c,int d){
        memset(temp,0,sizeof temp);
        for (int i = 1;i <= 20;i++){
            temp[i] += pre[1][a][i];
            temp[i] += pre[2][b][i];
            temp[i] += pre[3][c][i];
            temp[i] += pre[4][d][i];
        }
        int num = 0,num1 = 0;
        for (int i = 1;i <= 20;i++){
            num+=temp[i]/2;
            temp[i]%=2;
            num1+=temp[i];
        }
        return make_pair(num1,num);
    }
    
    int main(){
        //freopen("F:\rush.txt","r",stdin);
        while (~scanf("%d",&n) && n){
            for (int i = n;i >= 1;i--)
                for (int j = 1;j <= 4;j++)
                    scanf("%d",&x[j][i]);
    
            for (int i = 1;i <= 4;i++){
                for (int k = 1;k <= 20;k++) pre[i][n+1][k] = 0;
                for (int j = n;j >= 1;j--){
                    for (int k = 1;k <= 20;k++)
                        pre[i][j][k] = pre[i][j+1][k];
                    pre[i][j][x[i][j]]++;
                }
            }
    
            memset(f,-INF,sizeof f);
            f[n][n][n][n] = 0;
            int ans = 0;
            pair <int,int> pii;
            for (int i = n;i >= 0;i--)
                for (int j = n;j >= 0;j--)
                    for (int k = n;k >= 0;k--)
                        for (int l = n;l >= 0;l--)
                            if (f[i][j][k][l]>=0){
                                ans = max(ans,f[i][j][k][l]);
                                if (i){
                                    pii = tongji(i,j+1,k+1,l+1);
                                    if (pii.first<5){
                                        int &t = f[i-1][j][k][l];
                                        t = max(t,pii.second);
                                    }
                                }
                                if (j){
                                    pii = tongji(i+1,j,k+1,l+1);
                                    if (pii.first<5){
                                        int &t = f[i][j-1][k][l];
                                        t = max(t,pii.second);
                                    }
                                }
                                if (k){
                                    pii = tongji(i+1,j+1,k,l+1);
                                    if (pii.first<5){
                                        int &t = f[i][j][k-1][l];
                                        t = max(t,pii.second);
                                    }
                                }
                                if (l){
                                    pii = tongji(i+1,j+1,k+1,l);
                                    if (pii.first<5){
                                        int &t = f[i][j][k][l-1];
                                        t = max(t,pii.second);
                                    }
                                }
                            }
            printf("%d
    ",ans);
        }
        return 0;
    }
  • 相关阅读:
    我在项目内使用了设计模式后,同事直呼看不懂
    pom文件中依赖找不到的根本解决方法
    基于session的传统认证授权详解
    python中2个字典比较
    编码设计应遵循的规则
    yarn任务执行流程
    python3 中print 显示不全问题
    pandas 可视化
    python时间大小判断,相差天数秒数计算
    Impala任务程序cancle
  • 原文地址:https://www.cnblogs.com/AWCXV/p/7626175.html
Copyright © 2011-2022 走看看