zoukankan      html  css  js  c++  java
  • POJ_1185_状态压缩dp

    http://poj.org/problem?id=1185

    一次考虑两行,比一行略为复杂。sta保存每种状态炮兵位置,sum保存每种状态当行炮兵总数,a保存地形,dp[i][j][k]表示到第i行当前行j状态上一行k状态的最大炮兵数。

    #include<iostream>
    #include<cstdio>
    #include<cstring>
    using namespace std;
    //int main()
    //{
    //    int num = 0;
    //    for(int i = 0;i <= (1<<10);i++)
    //    {
    //        if(i & (i<<2) || i & (i<<1))  continue;
    //        num++;
    //    }
    //    printf("%d",num);
    //}
    
    int dp[105][65][65],a[105] = {0},n,m,sta[65],sum[65] = {0};
    
    int main()
    {
        memset(dp,-1,sizeof(dp));
        scanf("%d%d",&n,&m);
        getchar();
        if(n == 0 && m == 0)
        {
            printf("0
    ");
            return 0;
        }
        for(int i = 0;i < n;i++)
        {
            char temp[15];
            gets(temp);
            for(int j = 0;j < m;j++)
            {
                if(temp[j] == 'H')  a[i] |= (1<<j);
            }
        }
    //    for(int i = 0;i < n;i++)    printf("%d
    ",a[i]);
        int cnt = 0;
        for(int i = 0;i < (1<<m);i++)
        {
            if(i & (i<<2))  continue;
            if(i & (i<<1))  continue;
            sta[cnt] = i;
            int temp = i;
            while(temp)
            {
                sum[cnt] += temp%2;
                temp /= 2;
            }
            cnt++;
        }
    //    for(int i = 0;i < cnt;i++)    printf("%d %d
    ",sta[i],sum[i]);
        for(int i = 0;i < cnt;i++)
        {
            if(a[0] & sta[i])   continue;
            dp[0][i][0] = sum[i];
        }
    //    for(int i = 0;i < cnt;i++)    printf("%d
    ",dp[0][i][0]);
        for(int i = 0;i < cnt;i++)
        {
            if(a[1] & sta[i])   continue;
            for(int j = 0;j < cnt;j++)
            {
                if(sta[i] & sta[j]) continue;
                dp[1][i][j] = max(dp[1][i][j],dp[0][j][0]+sum[i]);
            }
        }
        for(int i = 2;i < n;i++)
        {
            for(int j = 0;j < cnt;j++)
            {
                if(a[i] & sta[j])   continue;
                for(int k = 0;k < cnt;k++)
                {
                    if(sta[j] & sta[k])   continue;
                    for(int l = 0;l < cnt;l++)
                    {
                        if(sta[j] & sta[l]) continue;
                        if(dp[i-1][k][l] == -1) continue;
                        dp[i][j][k] = max(dp[i][j][k],dp[i-1][k][l]+sum[j]);
                    }
                }
            }
        }
    //    for(int i = 0;i < cnt;i++)
    //    {
    //        for(int j = 0;j < cnt;j++)  printf("%d ",dp[n-1][i][j]);
    //        printf("
    ");
    //    }
        int ans = -1;
        for(int i = 0;i < cnt;i++)
        {
            for(int j = 0;j < cnt;j++)  ans = max(ans,dp[n-1][i][j]);
        }
        printf("%d
    ",ans);
        return 0;
    }
  • 相关阅读:
    网站页面性能优化的 34条黄金守则 (雅虎团队经验)
    进程调度算法小结
    玩转TCP连接
    数据包在网络中的流转
    浅入理解JVM虚拟机
    Leecode no.47 全排列 II
    Leecode no.143 重排链表
    关于我用设计模式对公司代码重构的这件事
    进程间通信方式小结
    Leecode no.82 删除排序链表中的重复元素 II
  • 原文地址:https://www.cnblogs.com/zhurb/p/5914324.html
Copyright © 2011-2022 走看看