• 作者:zifeiy
  • 标签:状态压缩、枚举、动态规划

题目链接:https://www.luogu.org/problem/P2258

这道题目状态压缩是肯定的,我们需要用二进制来枚举状态。

江湖上有一句话,叫做“暴力出奇迹”,所以我一开始是暴力枚举的。

暴力枚举50分

下面是我暴力枚举(骗分50)的思路(后续动态规划的思想也是建立在此基础之上,所以最好还是了解一下)。

首先用二进制枚举所有选择r行的行的排列,然后用二进制枚举所有选择c列的排列,然后计算选中了这r行c列的结果,与最终答案比较。

时间复杂度为 \(O( 2^n \times 2^m \times n \times m ) = O(2^{40})\) ,会超时,过了50%数据。

实现代码如下:

#include <bits/stdc++.h>
using namespace std;
const int maxn = 22;
int n, m, r, c, a[maxn][maxn], b[maxn][maxn], ans = -1;
// 枚举行和列,但是这样只能过50%数据,另50%数据会TLE
void handle() {
int tmp = 0;
for (int i = 0; i < r; i ++) for (int j = 0; j < c; j ++) {
if (i) tmp += abs(b[i][j] - b[i-1][j]);
if (j) tmp += abs(b[i][j] - b[i][j-1]);
}
if (ans == -1 || ans > tmp) {
ans = tmp;
}
}
int main() {
scanf("%d%d%d%d", &n, &m, &r, &c);
for (int i = 0; i < n; i ++) for (int j = 0; j < m; j ++) scanf("%d", &a[i][j]);
for (int s1 = 0; s1 < (1<<n); s1 ++) {
if (__builtin_popcount(s1) != r) continue;
for (int s2 = 0; s2 < (1<<m); s2 ++) {
if (__builtin_popcount(s2) != c) continue;
int id1 = 0;
for (int i = 0; i < n; i ++) {
if (s1 & (1<<i)) {
int id2 = 0;
for (int j = 0; j < m; j ++) {
if (s2 & (1<<j)) {
b[id1][id2++] = a[i][j];
}
}
id1 ++;
}
}
handle();
}
}
printf("%d\n", ans);
return 0;
}

枚举+DP100分

首先还是二进制枚举所有选择了r行的排列,然后对于选中的r行,我们定义状态 \(f[i][j]\) 表示到第\(i\)列(且包含第\(i\)列)并且此时选择了 \(j\) 列的最小值。

则我们只需要遍历每一列 \(i\) ,对于列号 \(i\) :

  • \(f[i][0] = 0\) ;
  • \(f[i][1] = t1\) (这里的 \(t1\) 就是选中的r行中元素在第 \(j\) 列的所有元素两两绝对值之和);

然后我们再从 \(0\) 到 \(i-1\) 遍历列号 \(j\) ,再从 \(2\) 到 \(\min (c, i+1)\) 遍历 \(k\) ,有状态转移方程:

\(f[i][k] = min(f[i][k], f[j][k-1] + t1 + t2)\) 。(其中 \(t2\) 表示第 \(i\) 列 和第 \(j\) 列 r对元素绝对值之差的和)。

这样,时间复杂度降到了 \(O( 2^n \times n^3 ) = O( 2^28 )\) ,这样能够过所有100%数据。

实现代码如下:

#include <bits/stdc++.h>
using namespace std;
const int maxn = 22;
const int INF = (1<<29);
int n, m, r, c, a[maxn][maxn], id[maxn], f[maxn][maxn], ans = -1;
void handle() {
for (int i = 0; i < m; i ++) for (int j = 2; j <= c; j ++) f[i][j] = INF;
for (int i = 0; i < m; i ++) {
int t1 = 0;
for (int j = 1; j < r; j ++) t1 += abs(a[ id[j-1] ][i] - a[ id[j] ][i]);
f[i][0] = 0;
f[i][1] = t1;
for (int j = 0; j < i; j ++) {
int t2 = 0;
for (int k = 0; k < r; k ++) t2 += abs(a[ id[k] ][j] - a[ id[k] ][i]);
for (int k = 2; k <= i+1 && k <= c; k ++)
f[i][k] = min(f[i][k], f[j][k-1] + t1 + t2);
} }
for (int i = c-1; i < m; i ++)
if (ans == -1 || ans > f[i][c]) ans = f[i][c];
}
int main() {
scanf("%d%d%d%d", &n, &m, &r, &c);
for (int i = 0; i < n; i ++) for (int j = 0; j < m; j ++) scanf("%d", &a[i][j]);
for (int s = 0; s < (1<<n); s ++) {
if (__builtin_popcount(s) != r) continue;
int cnt = 0;
for (int i = 0; i < n; i ++) if (s & (1<<i)) id[cnt++] = i;
handle();
}
printf("%d\n", ans);
return 0;
}

洛谷P2258 子矩阵 题解 状态压缩/枚举/动态规划的更多相关文章

  1. 洛谷P2258 子矩阵——题解

    题目传送 表示一开始也是一脸懵逼,虽然想到了DP,但面对多变的状态不知从何转移及怎么合理记录状态.之(借鉴大佬思路)后,豁然开朗,于是在AC后分享一下题解. 发现数据范围出奇地小,不过越是小的数据范围 ...

  2. 洛谷 P2258 子矩阵 解题报告

    P2258 子矩阵 题目描述 给出如下定义: 子矩阵:从一个矩阵当中选取某些行和某些列交叉位置所组成的新矩阵(保持行与列的相对顺序)被称为原矩阵的一个子矩阵. 例如,下面左图中选取第 2 . 4行和第 ...

  3. 洛谷P2258 子矩阵

    P2258 子矩阵 题目描述 给出如下定义: 子矩阵:从一个矩阵当中选取某些行和某些列交叉位置所组成的新矩阵(保持行与列的相对顺序)被称为原矩阵的一个子矩阵. 例如,下面左图中选取第2.4行和第2.4 ...

  4. 洛谷 P2258 子矩阵

    题目描述 给出如下定义: 子矩阵:从一个矩阵当中选取某些行和某些列交叉位置所组成的新矩阵(保持行与列的相对顺序)被称为原矩阵的一个子矩阵. 例如,下面左图中选取第2.4行和第2.4.5列交叉位置的元素 ...

  5. 【Luogu】P2258子矩阵(状态压缩,DP)

    233今天蒟蒻我连文化课都没听光想着这个了 然后我调了一下午终于过了!!! 一看数据范围似乎是状压,然而216等于65536.开一个65536*65536的二维数组似乎不太现实. 所以Rqy在四月还是 ...

  6. 洛谷P2258 子矩阵[2017年5月计划 清北学堂51精英班Day1]

    题目描述 给出如下定义: 子矩阵:从一个矩阵当中选取某些行和某些列交叉位置所组成的新矩阵(保持行与列的相对顺序)被称为原矩阵的一个子矩阵. 例如,下面左图中选取第2.4行和第2.4.5列交叉位置的元素 ...

  7. 洛谷P1357 花园(状态压缩 + 矩阵快速幂加速递推)

    题目链接:传送门 题目: 题目描述 小L有一座环形花园,沿花园的顺时针方向,他把各个花圃编号为1~N(<=N<=^).他的环形花园每天都会换一个新花样,但他的花园都不外乎一个规则,任意相邻 ...

  8. 洛谷P1036 选数 题解 简单搜索/简单状态压缩枚举

    题目链接:https://www.luogu.com.cn/problem/P1036 题目描述 已知 \(n\) 个整数 \(x_1,x_2,-,x_n\) ,以及 \(1\) 个整数 \(k(k& ...

  9. UVA 1508 - Equipment 状态压缩 枚举子集 dfs

    UVA 1508 - Equipment 状态压缩 枚举子集 dfs ACM 题目地址:option=com_onlinejudge&Itemid=8&category=457& ...

随机推荐

  1. QT生成GUID

    #include <QCoreApplication> #include <QUuid> #include <QDebug> int main(int argc, ...

  2. 第十章—DOM(二)——Element类型

    Element类型用于表现HTML和XML,提供了对元素标签名,子节点和特效的访问.Element节点具有以下特征: 要访问元素的标签名,可以使用nodeName属性,也可以使用tagName属性.这 ...

  3. arcgis地图窗口操作

    <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title> ...

  4. ASP.NET+C#面试题

    1.维护数据库的完整性.一致性.你喜欢用触发器还是自写业务逻辑?为什么? 尽可能用约束(包括CHECK.主键.唯一键.外键.非空字段)实现,这种方式的效率最好:其次用触发器,这种方式可以保证无论何种业 ...

  5. Floyd算法模板--详解

    对于无权的图来说: 若从一顶点到另一顶点存在着一条路径,则称该路径长度为该路径上所经过的边的数目,它等于该路径上的顶点数减1. 由于从一顶点到另一顶点可能存在着多条路径,每条路径上所经过的边数可能不同 ...

  6. 2-4 Numpy+Matplotlib可视化(二)

    自定义绘图 # -*-coding:utf-8-*- # !/usr/bin/env python # Author:@vilicute import numpy as np import matpl ...

  7. 【滴水石穿】rn

    这个项目还不错,还比较全 先放项目地址:https://github.com/ShionHXC/rn 项目算是一个完整的APP 有用到redux-thunk存储数据,算的上是一个普通的比较完整的APP ...

  8. python中的open函数

    open函数用于文件处理 操作文件时,一般需要经历如下步骤: 打开文件 操作文件 一.打开文件 1 文件句柄 = open('文件路径', '模式') 打开文件时,需要指定文件路径和以何等方式打开文件 ...

  9. JavaSript中的正则表达式

    正则表达式是对字符串操作的逻辑公式,表达了对字符串的一种过滤逻辑. 相对于.NET和Perl,JS对正则表达式的支持相当朴素,或者说JS的正则表达式是perl正则表达式的一个子集. 一.正则表达式引擎 ...

  10. iOS9新系统下App Store应用上传新指南

    http://www.cocoachina.com/appstore/20151010/13691.html 最近频繁收到小伙伴们的反馈,说经过前期学习已对ASO规则略有了解,但APP的提交审核是由技 ...