zoukankan      html  css  js  c++  java
  • hdu 1565 方格取数(1)(状态压缩dp)

    方格取数(1)

                                                                    Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)

    Problem Description
    给你一个n*n的格子的棋盘,每一个格子里面有一个非负数。
    从中取出若干个数,使得随意的两个数所在的格子没有公共边。就是说所取的数所在的2个格子不能相邻,而且取出的数的和最大。
     

    Input
    包含多个測试实例,每一个測试实例包含一个整数n 和n*n个非负数(n<=20)
     

    Output
    对于每一个測试实例,输出可能取得的最大的和
     

    Sample Input
    3 75 15 21 75 15 28 34 70 5
     

    Sample Output
    188
     


    分析:dp[i][j]表示前i行,第i行取第j个状态时的取值总和。则dp[i][j] = max(dp[i][j], dp[i-1][k] + sum[i][j]).当中sum[i][j]表示第i行取第j个状态的取值总和。

    由于n<=20。经计算能够发现,合法状态最多有17711个。

    #include <cstdio>
    #include <cstring>
    #include <algorithm>
    using namespace std;
    
    const int N = 21;
    const int M = 17720; //合法状态最多有17711个
    int n, p;
    int a[N][N];
    int dp[N][M];
    int s[M];
    int sum[N][M];
    
    bool checkA(int x) {  //推断本行状态是否冲突
        return !(x & (x >> 1));
    }
    
    bool checkB(int x, int y) { //推断本行和上一行是否冲突
        return !(x & y);
    }
    
    int get_sum(int r, int state) {  //求第i行状态为state时的取值总和
        int res = 0;
        for(int i = 0; i < n; i++)
            if((state >> i) & 1)
                res += a[r][n - 1 - i];
        return res;
    }
    
    void Init() {
        p = 0;
        memset(sum, 0, sizeof(sum));
    
        for(int i = 0; i < (1 << n); ++i) //求出全部合法状态
            if(checkA(i))
                s[p++] = i;
    
        for(int i = 0; i < n; ++i) {  //求第i行取第j个状态时的取值总和
            for(int j = 0; j < p; ++j) {
                sum[i][j] = get_sum(i, s[j]);
            }
        }
    }
    
    void solve() {
        memset(dp, 0, sizeof(dp));
        for(int i = 0; i < p; i++)
            dp[0][i] = sum[0][i];
    
        for(int i = 1; i < n; i++) { //行数
            for(int j = 0; j < p; j++) { //本行状态
                for(int k = 0; k < p; k++) { //上一行的状态
                    if(checkB(s[j], s[k])) {
                        dp[i][j] = max(dp[i][j], dp[i-1][k] + sum[i][j]);
                    }
                }
            }
        }
    
        int ans = dp[n-1][0];
        for(int i = 1; i < p; i++)
            if(ans < dp[n-1][i])
                ans = dp[n-1][i];
        printf("%d
    ", ans);
    }
    
    int main() {
        while(~scanf("%d", &n)) {
            for(int i = 0; i < n; i++)
                for(int j = 0; j < n; j++)
                    scanf("%d", &a[i][j]);
            Init();
            solve();
        }
        return 0;
    }


  • 相关阅读:
    错误 : 资产文件“项目objproject.assets.json”没有“.NETCoreApp,Version=v2.0”的目标。确保已运行还原,且“netcoreapp2.0”已包含在项目的 TargetFrameworks 中。
    .NET core RSA帮助类
    vs2010单步调试崩溃
    常用排序算法比较与分析
    js原生的轮播图
    vue-cli脚手架使用-- 初学者
    vue-cli配置基础
    jq轮播图插件
    bootstrap 获得轮播中的索引或当前活动的焦点对象
    jQuery学习笔记之Ajax用法详解
  • 原文地址:https://www.cnblogs.com/mthoutai/p/7141478.html
Copyright © 2011-2022 走看看