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

题目传送门


题目描述

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 can’t be planted. Canny FJ knows that the cows dislike eating close to each other, so when choosing which squares to plant, he avoids choosing squares that are adjacent; no two chosen squares share an edge. He has not yet made the final choice as to which squares to plant.

Being a very open-minded man, Farmer John wants to consider all possible options for how to choose the squares for planting. He is so open-minded that he considers choosing no squares as a valid option! Please help Farmer John determine the number of ways he can choose the squares to plant.

农场主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


题解

  • 数据范围很明显就是状态压缩了
  • f[ i ] 记录每行允许的状态,预处理一下
  • 再把所有合法的状态标记一下:state数组(没有相邻的 1 即为合法状态)
  • 0 ~ (1 << n) - 1这些状态中寻找合法的状态:
    1、没有相邻的1
    2、是 f[ i ]的子集
  • 然后找上一行的合法状态(上下两行之间没有相邻的草地:k & j == 0)加到 dp[i][j] 即可
  • 最后跑一边,把最后一行的各个状态的值求个和就是答案

AC-Code

#include <bits/stdc++.h>

using namespace std;
typedef long long ll;
#define ios ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);

const int mod = 1e9;
int f[13]; // 第 i 行的土地允许状态
int dp[13][1 << 12]; // 在前 i 行,第 i 行状态为 j 时的最大方案数
bool state[1 << 12]; // 记录 i 这个状态是否合法,即不允许有相邻的 1
int main() {
	ios;
	int n, m;
	while (cin >> n >> m) {
		for (int i = 1; i <= n; i++) {
			int sum = 0;
			for (int j = 1; j <= m; j++) {
				int x;
				cin >> x;
				sum = (sum << 1) + x;
			}
			f[i] = sum;
		}
		for (int i = 0; i < 1 << m; i++) // 不允许有相邻的 1
			state[i] = !(i & (i << 1L)); // 这里不需要左移、右移都判断,只需要用1L来防止左移溢出即可
		memset(dp, 0, sizeof(dp));
		dp[0][0] = 1;

		for (int i = 1; i <= n; i++)
			for (int j = 0; j < 1 << m; j++)
				if (state[j] && ((j & f[i]) == j)) // 保证合法的 j ,并且 j 是 f[i] 的子集;
												   // 也可以写成(j | f[i]) == f[i]
					for (int k = 0; k < 1 << m; k++)
						if (!(k & j))
							dp[i][j] = (dp[i][j] + dp[i - 1][k]) % mod;

		/* 或者
		for (int i = 1; i <= n; i++) {
			int j;
			for ( j = f[i]; j; j = (j - 1) & f[i])	//枚举 f[i] 的非空子集
				if (state[j])
					for (int k = 0; k < 1 << m; k++)
						if (!(k & j))
							dp[i][j] = (dp[i][j] + dp[i - 1][k]) % mod;
			j = 0;	// 额外对空集处理一下
			if (state[j])
				for (int k = 0; k < 1 << m; k++)
					if (!(k & j))
						dp[i][j] = (dp[i][j] + dp[i - 1][k]) % mod;
		}
		*/
		int ans = 0;
		for (int i = 0; i < 1 << m; i++) {
			ans += dp[n][i];
			ans %= mod;
		}
		cout << ans << endl;
	}
}

题目描述 农夫约翰一直在观察他的奶牛们。他注意到,如果在牛群中有太多的牛靠得太近,就会导致不健康的行为和情感问题。 约翰想知道他的牛群是否存在这个问题。他定义这个问题为:在一个固定长度的路段上,如果有两头高度大于等于 $y$ 的奶牛之间的距离小于 $x$,则牛群中就存在一个挤得太近的情况。 约翰有 $N$ 头牛 ($1 \leq N \leq 50,000$),每头牛的高度为 $h_i$ ($1 \leq h_i \leq 1,000,000$)。他想知道是否存在一对牛,使得它们之间的距离小于 $x$,且它们的高度都大于等于 $y$。 输入格式 第一行包含三个整数 $N, L, R$,分别表示牛的数量,路段长度,和问题的最大高度。 接下来 $N$ 行,每行一个整数 $h_i$,表示每头牛的高度。 输出格式 如果存在一对牛,它们之间的距离小于 $x$,且它们的高度都大于等于 $y$,则输出 $1$,否则输出 $0$。 输入样例1 4 6 4 4 4 5 7 输出样例1 1 输入样例2 5 3 3 1 5 5 5 5 输出样例2 0 提示 对于 $30\%$ 的数据,$N \leq 500$。 对于 $100\%$ 的数据,$1 \leq N \leq 50,000$,$1 \leq L \leq 1,000,000$,且 $L \leq R$。 数据范围 时间限制:1.0s,空间限制:256MB 算法1 (暴力枚举) $O(n^2)$ 首先对输入的牛的高度进行排序,之后枚举每头牛,再枚举它后面的每头牛,如果两头牛的高度均大于等于 $y$,且它们之间的距离小于 $x$,则输出 $1$。如果最后仍然没有满足条件的牛,则输出 $0$。 时间复杂度 暴力枚举,时间复杂度为 $O(n^2)$,无法通过此题。 算法2 (滑动窗口) $O(n \log n)$ 为了方便后续操作,我们将所有的牛按照它们的高度从小到大排序。之后,我们维护一个长度为 $L$ 的滑动窗口,它的右端点与左端点之间的距离小于 $x$。我们从左到右扫描每头牛,将它加入滑动窗口的左端点,同时将滑动窗口右移,直到滑动窗口的右端点与左端点之间的距离小于 $x$。 在处理完一头牛之后,我们需要判断滑动窗口中是否存在一对牛,它们的高度均大于等于 $y$,且它们之间的距离小于 $x$。我们可以用双指针来实现这个操作。我们从滑动窗口的左端点开始,向右移动一个指针 $i$,同时向右移动一个指针 $j$,直到 $h_j - h_i \leq x$。在这个过程中,我们需要判断 $h_i$ 和 $h_j$ 是否均大于等于 $y$。如果存在一对牛满足条件,则输出 $1$。如果最后仍然没有满足条件的牛,则输出 $0$。 时间复杂度 因为需要对所有的牛进行排序,所以时间复杂度为 $O(n \log n)$。 C++ 代码 算法3 (暴力优化) $O(n \log n)$ 首先对输入的牛的高度进行排序,之后枚举每头牛。如果当前牛的高度小于 $y$,则跳过这头牛。否则,我们从它的左边和右边各扩展出一个长度为 $x$ 的区间。如果这两个区间内的牛的数量均大于等于 $2$,且这两个区间中任意两头牛的高度均大于等于 $y$,则输出 $1$。 时间复杂度 因为需要对所有的牛进行排序,所以时间复杂度为 $O(n \log n)$。 C++ 代码
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值