zoukankan      html  css  js  c++  java
  • Corn Fields POJ

    第一个题题目链接: POJ - 3254 

    第二个题题目链接:POJ - 1185 

    第一个题的题目大意:给你一个n*m的01矩阵,然后让你安排奶牛,只有为1的地方能安置奶牛,0的地方不能安置奶牛。当在一个为1的地方安置奶牛的时候,这个奶牛的四周都不能防止奶牛,会起冲突。然后问你一共有多少种安置方案。

    具体思路:我们把每一行按照二进制压缩,然后每一次枚举合法的状态,看是否能安置上奶牛,然后枚举当前一行的时候,判断是否冲突就好了。然后最终计算答案的时候累加最后一行为某一个合法状态时的方案数。

    AC代码:

     1 //dp[i][j]代表当前第i行为j时候的合法数。
     2 #include<iostream>
     3 #include<stdio.h>
     4 #include<cmath>
     5 #include<string>
     6 #include<cstring>
     7 #include<algorithm>
     8 using namespace std;
     9 # define ll long long
    10 # define inf 0x3f3f3f3f
    11 const int maxn = 5e3+100;
    12 const int mod  = 1e9;
    13 int dp[20][maxn];
    14 int a[20][20];
    15 int n,m;
    16 bool check(int t1,int t2)
    17 {
    18     if((((t2<<1)&t2)==0)&&(((t2>>1)&t2)==0))
    19     {
    20         for(int j=0; j<m; j++){
    21             if(((t2&(1<<j))&&a[t1][j])||(!(t2&(1<<j))))
    22                 continue;
    23             else
    24                 return false;
    25         }
    26         return true;
    27     }
    28     else
    29         return false;
    30 }
    31 int main()
    32 {
    33     scanf("%d %d",&n,&m);
    34     for(int i=0; i<n; i++)
    35     {
    36         for(int j=0; j<m; j++)
    37         {
    38             scanf("%d",&a[i][j]);
    39         }
    40     }
    41     int maxstate=(1<<m)-1;
    42     for(int i=0; i<n; i++)
    43     {
    44         if(i==0)
    45         {
    46             for(int j=0; j<=maxstate; j++)
    47             {
    48                 if(check(i,j))
    49                     dp[i][j]++;
    50             }
    51         }
    52         else
    53         {
    54             for(int j=0; j<=maxstate; j++)
    55             {
    56                 for(int k=0; k<=maxstate; k++)
    57                 {
    58                     if(check(i,k)&&((j&k)==0))
    59                     {
    60                         dp[i][k]+=dp[i-1][j];
    61                     }
    62                 }
    63             }
    64         }
    65     }
    66 //    for(int i=0;i<=maxstate;i++){
    67 //    cout<<i<<" "<<dp[0][i]<<endl;
    68 //    }
    69     ll sum=0;
    70     for(int i=0; i<=maxstate; i++)
    71     {
    72         sum=(sum+dp[n-1][i])%mod;
    73     }
    74     printf("%lld
    ",sum);
    75     return 0;
    76 }
    View Code

    第二个题题目大意:中文,和上一个题不同的是这个题求的是最大的合法状态。

    具体思路:这个题的难度比上一个题的难度增加了一点,上一个题dp[i][j]表示第i行选j这个状态时的合法数。但是这个题需要判断和上两行的关系。所以我们开一个三维数组。

    dp[i][j][k]代表第i行选择j状态,第i-1行选k这个状态时的合法数。然后求一个最大值就好了。

    具体判断的时候,如果当前的这一行和sto[i],a[i]&sto[j]==sto[j]的时候,就可以了。

    注意需要先预处理出合法的状态,否则会爆内存,因为是个三维空间。

    AC代码:

      1 #include<iostream>
      2 #include<stdio.h>
      3 #include<cmath>
      4 #include<string>
      5 #include<algorithm>
      6 #include<vector>
      7 #include<vector>
      8 using namespace std;
      9 # define ll long long
     10 # define inf 0x3f3f3f3f
     11 const int maxn = 2e2+100;
     12 const int mod  = 1e9;
     13 int dp[100+10][maxn][maxn];
     14 int a[200];
     15 char str[100+100][20];
     16 int sto[maxn],num[maxn];
     17 int n,m;
     18 bool check(int t)
     19 {
     20     if(((t<<1)&t)||((t>>1)&t)||((t<<2)&t)||((t>>2)&t))
     21         return false ;
     22     return true;
     23 }
     24 int cal(int t)
     25 {
     26     int ans=0;
     27     while(t)
     28     {
     29         ans+=(t&1);
     30         t>>=1;
     31     }
     32     return ans;
     33 }
     34 int main()
     35 {
     36     scanf("%d %d",&n,&m);
     37     for(int i=0; i<n; i++)
     38     {
     39         scanf("%s",str[i]);
     40         for(int j=0; j<m; j++)
     41         {
     42             a[i]<<=1;
     43             a[i]|=(str[i][j]=='P'?1:0);
     44         }
     45     }
     46     int cnt=0;
     47     int maxstate=(1<<m)-1;
     48     for(int i=0; i<=maxstate; i++)
     49     {
     50         if(check(i))
     51         {
     52             sto[++cnt]=i;
     53             num[cnt]=cal(i);
     54         }
     55     }
     56     int maxx=0;
     57     for(int i=0; i<n; i++)
     58     {
     59         if(i==0)
     60         {
     61             for(int j=1; j<=cnt; j++)
     62             {
     63                 if((a[i]&sto[j])!=sto[j])continue;
     64                     dp[i][j][0]=max(dp[i][j][0],num[j]);
     65                     maxx=max(maxx,dp[i][j][0]);
     66             }
     67         }
     68         else if(i==1)
     69         {
     70             for(int j=1; j<=cnt; j++)
     71             {
     72                 if((sto[j]&a[i])!=sto[j])continue;
     73                 for(int k=1; k<=cnt; k++)
     74                 {
     75                     if((sto[k]&a[i-1])!=sto[k])continue;
     76                     if((sto[j]&sto[k])!=0)continue;
     77                     dp[i][j][k]=max(dp[i][j][k],dp[i-1][k][0]+num[j]);
     78                     maxx=max(maxx,dp[i][j][k]);
     79                 }
     80             }
     81         }
     82         else
     83         {
     84             for(int j=1; j<=cnt; j++)
     85             {
     86                 if((sto[j]&a[i])!=sto[j])
     87                     continue;
     88                 for(int k=1; k<=cnt; k++)
     89                 {
     90                     if((sto[k]&a[i-1])!=sto[k]||(sto[k]&sto[j])!=0)
     91                         continue;
     92                     for(int L=1; L<=cnt; L++)
     93                     {
     94                         if((sto[L]&a[i-2])!=sto[L])
     95                             continue;
     96                         if((sto[j]&sto[L])!=0||(sto[k]&sto[L])!=0)
     97                             continue;
     98                         dp[i][j][k]=max(dp[i][j][k],dp[i-1][k][L]+num[j]);
     99                         maxx=max(maxx,dp[i][j][k]);
    100                     }
    101                 }
    102             }
    103         }
    104     }
    105     printf("%d
    ",maxx);
    106 }
    View Code
  • 相关阅读:
    函数进阶-生成器
    函数进阶-列表生成式
    闭包
    命名空间
    内置方法
    函数
    squid清除缓存
    subprocess实现管道
    Python统计脚本行数(fileinput)
    fabric note
  • 原文地址:https://www.cnblogs.com/letlifestop/p/10807760.html
Copyright © 2011-2022 走看看