玉米田(cowfood)

来源:互联网 发布:java二维数组意思 编辑:程序博客网 时间:2024/05/16 06:49

题意/Description:

       农民 John 购买了一处肥沃的矩形牧场,分成M*N(1 <= M <= 12; 1 <= N <= 12)个格子。他想在那里的一些格子中种植美味的玉米。遗憾的是,有些格子区域的土地是贫瘠的,不能耕种。精明的 FJ 知道奶牛们进食时不喜欢和别的牛相邻,所以一旦在一个格子中种植玉米,那么他就不会在相邻的格子中种植,即没有两个被选中的格子拥有公共边。他还没有最终确定哪些格子要选择种植玉米。

       作为一个思想开明的人,农民 John 希望考虑所有可行的选择格子种植方案。由于太开明,他还考虑一个格子都不选择的种植方案!请帮助农民 John 确定种植方案总数。


读入/Input

       Line 1: 两个用空格分隔的整数 M 和 N
  Lines 2..M+1: 第 i+1 行描述牧场第i行每个格子的情况, N 个用空格分隔的整数,表示 这个格子是否可以种植(1 表示肥沃的、适合种植,0 表示贫瘠的、不可种植)


输出/Output

       Line 1: 一个整数: FJ 可选择的方案总数 除以 100,000,000 的余数。


题解/solution

       看到n,m<=12,可以考虑一下状压DP。

       因为M,N <=12 所以可以用一个longint来表示一行的状态。
       就可以用 f[i][nm] 表示第i行,状态是nm的时候的方案数。
       然后’2^12*2^12’枚举转移,判断一下是否可以转移。
       ∑dp[n][i]为答案。


代码/Code

var  n,m,nm,ans:longint;  a:array [0..13] of longint;  f:array [0..13,0..5001] of longint;procedure init;var  i,j,z:longint;begin  readln(n,m);  for i:=1 to n do    for j:=1 to m do      begin        read(z);        a[i]:=a[i] shl 1+z;      end;  nm:=(1 shl m)-1;end;procedure main;var  i,j,k:longint;begin  for i:=0 to nm do    if ((i and (i shr 1))=0) and (i or a[1]=a[1]) then      f[1,i]:=1;  for i:=2 to n do    for j:=0 to nm do      if f[i-1,j]<>0 then        for k:=0 to nm do          if (j and k=0) and (k or a[i]=a[i]) and (k and (k shr 1)=0) then            f[i,k]:=(f[i,k]+f[i-1,j]) mod 100000000;  for i:=0 to nm do    ans:=(ans+f[n,i]) mod 100000000;end;begin  init;  main;  write(ans);end.


2 0
原创粉丝点击