玉米田

内存限制:128 MiB

时间限制:1000 ms

标准输入输出

题目类型:传统

评测方式:文本比较

题目描述

农场主John新买了一块长方形的新牧场,这块牧场被划分成M行N列(1 ≤ M ≤ 12; 1 ≤ N ≤ 12),每一格都是一块正方形的土地。John打算在牧场上的某几格里种上美味的草,供他的奶牛们享用。

遗憾的是,有些土地相当贫瘠,不能用来种草。并且,奶牛们喜欢独占一块草地的感觉,于是John不会选择两块相邻的土地,也就是说,没有哪两块草地有公共边。

John想知道,如果不考虑草地的总块数,那么,一共有多少种种植方案可供他选择?(当然,把新牧场完全荒废也是一种方案)

输入格式

第一行:两个整数M和N,用空格隔开。

第2到第M+1行:每行包含N个用空格隔开的整数,描述了每块土地的状态。第i+1行描述了第i行的土地,所有整数均为0或1,是1的话,表示这块土地足够肥沃,0则表示这块土地不适合种草。

输出格式

一个整数,即牧场分配总方案数除以100,000,000的余数。

样例

样例输入

2 3

1 1 1

0 1 0

样例输出

9

(此题注意方案数要除以余数)

这道题很容易想到DP,并且是状压DP。为什么呢,设状态\(f_{i}\)为到第i行为止总的方案数,如果我们不能表示出\(i-1\)行的状态的话,那么第\(i\)行也推不出来。(第\(i\)行和第\(i-1\)行不能有土地邻接)

于是考虑状态压缩。用一个长度为\(m\)的二进制数\(j\)来表示第\(i\)行选择土地的情况,\(j\)的第\(k\)位为1表示在第\(i\)行的第\(k\)列选了一个土地。

于是我们便得到了状态转移方程:

\(f_{i,j} = \sum_{f_{i-1,k}}\)

那么这篇题解到这里就结束了

好吧别打我。

这道题的难点就是在于判断\(j\),\(k\),\(j\)和\(k\)这两个状态是否合法。这里就体现出了位运算的优越性。

对于判断一行农田\(j\)是否有相邻的已选的农田,我们可以这样判断:

if( (j<<1) & j)  不合法
else 合法

(位运算这篇博客里不会详讲,这种方法的可行性请读者自己翻阅资料思考)

同理对于状态\(k\)我们也可以这样判断。

对于上下相接的两个状态\(j\)和\(k\),我们要判断它们的每一位是否出现了都是1的情况:

if(i&j) 不合法
else 合法

(还有一个居然忘讲了)

需要注意的是,在枚举\(j\)的过程中,我们还要判断我们是否选择了“贫瘠”的土地,这是我使用的方法:

for(int i=1;i<=n;i++)  for(int j=1,L;j<=m;j++) {
cin>>L;
P[i]=(P[i]<<1)+L;
}//将初始的农田转换成2进制
for(int i=0;i<(1<<m);i++)
if((!((i<<1) & i)) && ((P[1] & i) == i))
f[1][i]=1; //初始化状态,((P[1] & i) == i)就是我用来判断是否
//选择了贫瘠土地的方法.

(关于状压DP,我这里要说一句,熟练掌握位运算是必要条件,这一点可以通过多做题来训练)

代码

#include <iostream>
#include <cstdio>
#include <vector>
using namespace std; #define N 15
#define LL long long
#define MOD 100000000 LL n,m,f[15][100010],ans,P[N]; int main() {
cin>>n>>m;
for(int i=1;i<=n;i++) for(int j=1,L;j<=m;j++) {
cin>>L;
P[i]=(P[i]<<1)+L;
}
for(int i=0;i<(1<<m);i++)
if((!((i<<1) & i)) && ((P[1] & i) == i))
f[1][i]=1;
for(int i=2;i<=n;i++)
for(int y=0;y<(1<<m);y++)
if((!((y<<1) & y)) && ((P[i] & y) == y))
for(int x=0;x<(1<<m);x++)
if((!((x<<1) & x)) && ((P[i-1] & x) == x)) {
if(x & y) continue;
f[i][y]=(f[i][y]+f[i-1][x])%MOD;
}
for(int i=0;i<(1<<m);i++)
if((!((i<<1) & i)) && ((P[n] & i) == i))
ans=(ans+f[n][i])%MOD;
cout<<ans;
}

状压DP : [USACO06NOV]玉米田的更多相关文章

  1. P1879 [USACO06NOV]玉米田Corn Fields(状压dp)

    P1879 [USACO06NOV]玉米田Corn Fields 状压dp水题 看到$n,m<=12$,肯定是状压鸭 先筛去所有不合法状态,蓝后用可行的状态跑一次dp就ok了 #include& ...

  2. P1879 [USACO06NOV]玉米田Corn Fields 状压dp/插头dp

    正解:状压dp/插头dp 解题报告: 链接! ……我真的太菜了……我以为一个小时前要搞完的题目调错误调了一个小时……90分到100我差不多搞了一个小时…… 然后这题还是做过的……就很气,觉得确实是要搞 ...

  3. [USACO06NOV]玉米田$Corn \ \ Fields$ (状压$DP$)

    #\(\mathcal{\color{red}{Description}}\) \(Link\) 农场主\(John\)新买了一块长方形的新牧场,这块牧场被划分成\(M\)行\(N\)列\((1 ≤ ...

  4. [USACO06NOV]玉米田Corn Fields 状压DP

    题面: 农场主John新买了一块长方形的新牧场,这块牧场被划分成M行N列(1 ≤ M ≤ 12; 1 ≤ N ≤ 12),每一格都是一块正方形的土地.John打算在牧场上的某几格里种上美味的草,供他的 ...

  5. 状压DP【洛谷P1879】 [USACO06NOV]玉米田Corn Fields

    P1879 [USACO06NOV]玉米田Corn Fields 农场主John新买了一块长方形的新牧场,这块牧场被划分成M行N列(1 ≤ M ≤ 12; 1 ≤ N ≤ 12),每一格都是一块正方形 ...

  6. [USACO06NOV]玉米田Corn Fields (状压$dp$)

    题目链接 Solution 状压 \(dp\) . \(f[i][j][k]\) 代表前 \(i\) 列中 , 已经安置 \(j\) 块草皮,且最后一位状态为 \(k\) . 同时多记录一个每一列中的 ...

  7. 状压DP【p1879】[USACO06NOV]玉米田Corn Fields

    Description 农场主John新买了一块长方形的新牧场,这块牧场被划分成M行N列(1 ≤ M ≤ 12; 1 ≤ N ≤ 12),每一格都是一块正方形的土地.John打算在牧场上的某几格里种上 ...

  8. 洛谷P1879 [USACO06NOV]玉米田Corn Fields(状压dp)

    洛谷P1879 [USACO06NOV]玉米田Corn Fields \(f[i][j]\) 表示前 \(i\) 行且第 \(i\) 行状态为 \(j\) 的方案总数.\(j\) 的大小为 \(0 \ ...

  9. 状压dp之二之三 炮兵阵地/玉米田 By cellur925

    一.简单的状压dp 玉米田 题目描述 Farmer John has purchased a lush new rectangular pasture composed of M by N (1 ≤ ...

随机推荐

  1. Visual Studio 2017打包安装项目

    在我们用VS编好上位机后,就可以在自己电脑运行上位机,但是想其他人电脑运行上位机可能就行不通了,因为其他人电脑不一定有所需要的运行环境.这时我们就需要打包安装,把运行软件所需要的环境都打包在安装包里. ...

  2. 回溯---Permutations II

    47.Permutations II (Medium)](https://leetcode.com/problems/permutations-ii/description/) [1,1,2] hav ...

  3. django信号相关

    Django中提供了“信号调度”,用于在框架执行操作时解耦.通俗来讲,就是一些动作发生的时候,信号允许特定的发送者去提醒一些接受者. 1.Django内置信号 Model signals pre_in ...

  4. java 进销存管理 商户管理 库存管理 springmvc SSM 项目源码

    统介绍: 1.系统采用主流的 SSM 框架 jsp JSTL bootstrap html5 (PC浏览器使用) 2.springmvc +spring4.3.7+ mybaits3.3  SSM 普 ...

  5. Linux性能优化从入门到实战:10 内存篇:如何利用Buffer和Cache优化程序的运行效率?

    缓存命中率   缓存命中率,是指直接通过缓存获取数据的请求次数,占所有数据请求次数的百分比,可以衡量缓存使用的好坏.命中率越高,表示使用缓存带来的收益越高,应用程序的性能也就越好.   实际上,缓存是 ...

  6. 【改】DOS文件格式转UNIX文件格式

    windows中的文本文件的换行符是"\r\n",而linux中是"\n",dos格式文件传输到unix系统时,会在每行的结尾多一个^M,当然也有可能看不到,但 ...

  7. 11JSP基础

    1.Jsp基础 1.1 简介 Jsp,全称 Java Server Page java服务页面,能提供java服务的页面 jsp vs html html: 由html标签组成的,输出静态内容. js ...

  8. #10042. 「一本通 2.1 练习 8」收集雪花 || 离散化 || 双指针法 || C++ || LOJ

    题目:#10042. 「一本通 2.1 练习 8」收集雪花 看到网上没有这道题的题解,所以写一下. 要标记数字是否存在,看到x<=1e9,所以考虑用离散化,然后开一个last数组,last[i] ...

  9. 线程工具类 - Semaphore(信号量)

    Semaphore官方文档 一.使用信号量实现线程间的通信 /** * Demo:使用信号量实现线程间通信*/ public class SemaphoreDemo { public static v ...

  10. php连接access

    1.在控制面板中打开管理工具图标. 2.双击其中的数据源(ODBC)图标. 3.选择系统 DSN 选项卡. 4.点击系统 DSN 选项卡中的添加. 5.选择Microsoft Access Drive ...