状态压缩dp poj 3254 hdu5045

近来感觉状态压缩dp的强大性(灵活利用了二进制运算很关键)。。。于是做了俩提来看看。。毕竟队友是专业的dp,我只是管中窥豹下而已。。日后有机会再与之玩耍玩耍。。。ps:如果上天再给我一次机会,当年我愿意选择状态dp而不是网络流(只针对目前比赛出题潮流)

经典问题,不相邻/禁点方案数问题。poj3254

#include<iostream>
#include<cstdio>
using namespace std;
int n,m;
int dp[5000][15];
int yu[5000];
int numstate=0;
int fib[15];
void init()            //n行m列,状态一行推一行
{
    scanf("%d%d",&n,&m);
     int maxs=(1<<m);
    for(int i=0;i<maxs;i++)
    {
        if(i&(i<<1));                   //预处理掉每行相邻的状态(注意这里的右移用的好)
        else yu[numstate++]=i;
    }
    int tx;
    for(int i=0;i<n;i++)
    {
        for(int j=0;j<m;j++)
        {
            scanf("%d",&tx);
            if(tx==0)                         //fib【i】记录第i行禁止放的位子
             fib[i]=fib[i]|(1<<(m-j-1));
        }
    }
}
void jhh()
{
    for(int i=0;i<numstate;i++)       //首行
    {
        if((yu[i]&fib[0]));
        else dp[yu[i]][0]=1;
    }
    for(int i=1;i<n;i++)
    {
        for(int j=0;j<numstate;j++)          //到当前行该(合法)状态的种树 =上行合法状态之和
        {
            if(fib[i]&yu[j])continue;
             for(int k=0;k<numstate;k++)
             {
                 if(yu[j]&yu[k]||yu[k]&fib[i-1])continue;   //合法而且上下不冲突
                 dp[yu[j]][i]=(dp[yu[j]][i]+dp[yu[k]][i-1])%100000000;
             }
        }
    }
    int ans=0;
    for(int i=0;i<numstate;i++)
    {
        ans=(ans+dp[yu[i]][n-1])%100000000;
    }
    printf("%d\n",ans);
}
int main()
{
    init();
    jhh();
}

hdu 5045, 求哪种排列最优。把复杂度从n!--》2^n*n。

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
using namespace std;
int n,m;
double dp[1040][12];        //dp【state】【I】做到第i题的状态的目前最优情况。
int numstate=0;
double a[12][1005];
int main()
{
    int T;
    scanf("%d",&T);int cnt=1;
    while(T--)
    {
        scanf("%d%d",&n,&m);
        for(int i=1;i<=n;i++)
          for(int j=1;j<=m;j++)
             scanf("%lf",&a[i][j]);

        double sums=0;
        for(int ii=0;ii<m/n;ii++)         //分成n段处理
        {
            for(int yy=0;yy<1029;yy++)    //注意点double 型数组 除了0外不能用memset初始化
              for(int xx=0;xx<12;xx++)
                      dp[yy][xx]=-1;

            dp[0][0]=0;
           int maxst=1<<n;
           for(int i=0;i<n;i++)
            for(int j=0;j<maxst;j++)
            {
               if(dp[j][i]==-1)continue;
                for(int k=1,kk=n;k<maxst;k=k<<1,kk--)
                {
                     if((j&k)==0)                   //&优先级比比较运算符的低啊!!
                          dp[j|k][i+1]=max(dp[j|k][i+1],dp[j][i]+a[kk][i+1+n*ii]);
                }
            }
            sums=sums+dp[maxst-1][n];
        }
         for(int yy=0;yy<1029;yy++)
              for(int xx=0;xx<12;xx++)
                      dp[yy][xx]=-1;
            dp[0][0]=0;
           int maxst=1<<n;
           for(int i=0;i<m%n;i++)
            for(int j=0;j<maxst;j++)
            {
               if(dp[j][i]==-1)continue;
                for(int k=1,kk=n;k<maxst;k=k<<1,kk--)
                {
                     if((j&k)==0)
                      dp[j|k][i+1]=max(dp[j|k][i+1],dp[j][i]+a[kk][i+1+m/n*n]);
                }
            }
            double maxss=-1;
            for(int i=0;i<maxst;i++)
               maxss=max(maxss,dp[i][m%n]);
            sums=sums+maxss;
        printf("Case #%d: %.5lf\n",cnt++,sums);
    }
}
时间: 2024-12-09 19:31:00

状态压缩dp poj 3254 hdu5045的相关文章

(状态压缩dp) poj 3254

Corn Fields Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 8729   Accepted: 4651 Description Farmer John has purchased a lush new rectangular pasture composed of M by N (1 ≤ M ≤ 12; 1 ≤ N ≤ 12) square parcels. He wants to grow some yumm

(状态压缩DP) poj 2978

Colored stones Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 1734   Accepted: 819 Description You are given a row of m stones each of which has one of k different colors. What is the minimum number of stones you must remove so that no

状态压缩DP——POJ 2923

对应POJ题目:点击打开链接 Exponentiation Time Limit:1000MS     Memory Limit:65536KB     64bit IO Format:%I64d & %I64u Submit Status Practice POJ 2923 Description Emma and Eric are moving to their new house they bought after returning from their honeymoon. Fortu

(状态压缩DP) poj 2441

Arrange the Bulls Time Limit: 4000MS   Memory Limit: 65536K Total Submissions: 3709   Accepted: 1422 Description Farmer Johnson's Bulls love playing basketball very much. But none of them would like to play basketball with the other bulls because the

POJ 3254 Corn Fields 状态压缩DP (C++/Java)

http://poj.org/problem?id=3254 题目大意: 一个农民有n行m列的地方,每个格子用1代表可以种草地,而0不可以.放牛只能在有草地的,但是相邻的草地不能同时放牛, 问总共有多少种方法. 思路: 状态压缩的DP. 可以用二进制数字来表示放牧情况并判断该状态是否满足条件. 这题的限制条件有两个: 1.草地限制. 2.相邻限制. 对于草地限制,因为输入的时候1是可以种草地的. 以"11110"草地分析,就只有最后一个是不可以种草的.取反后得00001  .(为啥取反

POJ 3254 Corn Fields 状态压缩DP

题目链接:http://poj.org/problem?id=3254 思路:状态压缩DP,状态方程为dp[i][j] += (dp[i-1][k]) code: #include <stdio.h> #include <string.h> #define N 500 const int MOD = 100000000; int dp[15][N],ant[N],n,m,k,map[15]; bool ok(int x) { if(x&(x<<1))return

poj 3254 Corn Fields ,状态压缩DP

题目链接 题意: 一个矩阵里有很多格子,每个格子有两种状态,可以放牧和不可以放牧,可以放牧用1表示,否则用0表示,在这块牧场放牛,要求两个相邻的方格不能同时放牛,即牛与牛不能相邻.问有多少种放牛方案(一头牛都不放也是一种方案) state[i] 表示对于一行,保证不相邻的方案 状态:dp[i][ state[j] ]  在状态为state[j]时,到第i行符合条件的可以放牛的方案数 状态转移:dp[i][ state[j] ] =Sigma dp[i-1][state'] (state'为符合条

poj 3254 Corn Fields(状态压缩dp)

Description Farmer John has purchased a lush new rectangular pasture composed of M by N (1 ≤ M ≤ 12; 1 ≤ N ≤ 12) square parcels. He wants to grow some yummy corn for the cows on a number of squares. Regrettably, some of the squares are infertile and

POJ 3254 【状态压缩DP】

题意: 给一块n*m的田地,1代表肥沃,0代表贫瘠. 现在要求在肥沃的土地上种草,要求任何两个草都不能相邻. 问一共有多少种种草的方法. 种0棵草也是其中的一种方法. n和m都不大于12. 思路: 状态压缩DP,dp[i][j]代表在第i行状态j一共有多少种可能的种植方法. j是二进制转化而来的状态,0代表不种草,1代表种草. dp[i]只受到两个限制,即dp[i-1]的某种状态,和当前土地的贫瘠状况. 只要保证&操作之后重复的为0就可以了 最后输出sum(dp[n][1...w])(w代表一共