应该很容易想到暴力骗分。

我们考虑暴力\(dfs\)枚举所有行的选择,列的选择,每次跑一遍记下分值即可。

时间复杂度:\(O(C_n^r \times C_m^c \times r \times c)\)

可以水过\(60pts\)。

#include<bits/stdc++.h>
#define INF 1000000007
using namespace std;
inline int read(){
register int s=0,f=1;
register char ch=getchar();
while(!isdigit(ch)){if(ch=='-')f*=-1;ch=getchar();}
while(isdigit(ch))s=(s<<1)+(s<<3)+(ch^48),ch=getchar();
return s*f;
}
const int N=20;
int n,m,r,c,ans=INF;
int a[N][N],fh[N],fl[N];
int solve(){
int sum=0;
for(int i=1;i<=r;i++){
for(int j=1;j<c;j++){
sum+=abs(a[fh[i]][fl[j]]-a[fh[i]][fl[j+1]]);
}
}
for(int j=1;j<=c;j++){
for(int i=1;i<r;i++){
sum+=abs(a[fh[i]][fl[j]]-a[fh[i+1]][fl[j]]);
}
}
return sum;
}
void dfsl(int dep,int cnt){
if(cnt>c){ans=min(ans,solve());return;}
if(dep>m)return;
fl[cnt]=dep;dfsl(dep+1,cnt+1);
fl[cnt]=0;dfsl(dep+1,cnt);
}
void dfsh(int dep,int cnt){
if(cnt>r){dfsl(1,1);return;}
if(dep>n)return;
fh[cnt]=dep;dfsh(dep+1,cnt+1);
fh[cnt]=0;dfsh(dep+1,cnt);
}
int main(){
ios::sync_with_stdio(false);
n=read(),m=read(),r=read(),c=read();
for(int i=1;i<=n;i++){
for(int j=1;j<=m;j++){
a[i][j]=read();
}
}
dfsh(1,1);
cout<<ans<<endl;
return 0;
}

【算法分析】

我们依然先来枚举行的选择,于是接下来问题就转化成了:

  • 一个\(r \times m\)的矩阵选\(c\)列,使其分值最小。

显然,这是一个\(01\)选择列的问题,所以我们直接\(dp\)就好了。

设状态\(dp[i][j]\)表示当前选择第\(i\)列的同时已经选择了\(j\)列的最小分数。

接下来我们预处理好两个东西:

  • \(up[i]\)表示在当前行选择下第\(i\)列的总分值。

  • \(num[i][j]\)表示在当前行选择下连接第\(i\)列和第\(j\)列的分值。

那么对于第\(dp[i][j]\),我们只需再枚举一个第\(k\)列。当连接第\(k\)列和第\(i\)列时,价值即为本身第\(i\)列的分值\(up[i]\)和连接两列的分值\(num[i][k]\)。

于是不难推出:

\[dp[i][j]=min\{ dp[k][j-1]+up[i]+num[k][i] \}
\]

\[i \in [1,m],j\in[2,min(i,c)],k \in[1,i)
\]

边界处理:

\[dp=\{0x7f\},dp[i][1]=up[i]
\]

更新最小分值:

\[ans=min\{dp[i][c]\}
\]

时间复杂度:\(O(C_n^r \times m^2n)\)

#include<bits/stdc++.h>
#define INF 1000000007
using namespace std;
inline int read(){
register int s=0,f=1;
register char ch=getchar();
while(!isdigit(ch)){if(ch=='-')f*=-1;ch=getchar();}
while(isdigit(ch))s=(s<<1)+(s<<3)+(ch^48),ch=getchar();
return s*f;
}
const int N=20;
int n,m,r,c,ans=INF;
int a[N][N],f[N],up[N],num[N][N],dp[N][N];
void solve(){
memset(up,0,sizeof(up));
memset(num,0,sizeof(num));
for(int j=1;j<=m;j++){
for(int i=1;i<r;i++){
up[j]+=abs(a[f[i]][j]-a[f[i+1]][j]);
}
}
for(int i=1;i<=m;i++){
for(int j=i+1;j<=m;j++){
for(int k=1;k<=r;k++){
num[i][j]+=abs(a[f[k]][i]-a[f[k]][j]);
}
}
}
memset(dp,0x7f,sizeof(dp));
int tot=INF;
for(int i=1;i<=m;i++){
dp[i][1]=up[i];
for(int j=2;j<=min(i,c);j++){
for(int k=1;k<i;k++){
dp[i][j]=min(dp[i][j],dp[k][j-1]+up[i]+num[k][i]);
}
}
tot=min(tot,dp[i][c]);
}
ans=min(ans,tot);
}
void dfs(int dep,int cnt){
if(cnt>r){solve();return;}
if(dep>n)return;
f[cnt]=dep;dfs(dep+1,cnt+1);
f[cnt]=0;dfs(dep+1,cnt);
}
int main(){
ios::sync_with_stdio(false);
n=read(),m=read(),r=read(),c=read();
for(int i=1;i<=n;i++){
for(int j=1;j<=m;j++){
a[i][j]=read();
}
}
dfs(1,1);
cout<<ans<<endl;
return 0;
}

\(\operatorname{Update}\) \(\operatorname{On}\) \(\operatorname{2019.10.29}\)

题解 洛谷P2258 【子矩阵】的更多相关文章

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

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

  2. 洛谷P2258 子矩阵

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

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

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

  4. 洛谷P2258 子矩阵 题解 状态压缩/枚举/动态规划

    作者:zifeiy 标签:状态压缩.枚举.动态规划 题目链接:https://www.luogu.org/problem/P2258 这道题目状态压缩是肯定的,我们需要用二进制来枚举状态. 江湖上有一 ...

  5. 洛谷 P2258 子矩阵

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

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

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

  7. 题解 洛谷P5018【对称二叉树】(noip2018T4)

    \(noip2018\) \(T4\)题解 其实呢,我是觉得这题比\(T3\)水到不知道哪里去了 毕竟我比较菜,不大会\(dp\) 好了开始讲正事 这题其实考察的其实就是选手对D(大)F(法)S(师) ...

  8. 题解 洛谷 P3396 【哈希冲突】(根号分治)

    根号分治 前言 本题是一道讲解根号分治思想的论文题(然鹅我并没有找到论文),正 如论文中所说,根号算法--不仅是分块,根号分治利用的思想和分块像 似却又不同,某一篇洛谷日报中说过,分块算法实质上是一种 ...

  9. 题解-洛谷P5410 【模板】扩展 KMP(Z 函数)

    题面 洛谷P5410 [模板]扩展 KMP(Z 函数) 给定两个字符串 \(a,b\),要求出两个数组:\(b\) 的 \(z\) 函数数组 \(z\).\(b\) 与 \(a\) 的每一个后缀的 L ...

随机推荐

  1. 高级UI-滤镜和颜色通道

    滤镜在图片处理里面有很多的运用,尤其是相机使用了大量的滤镜,通过对颜色通道的调和,可以呈现出各种各样的效果 对图像进行一定的过滤加工处理,使用Paint设置滤镜效果 很多高级UI使用时候需要关闭硬件加 ...

  2. python爬虫2

    学习任务 获取去哪儿网的出发地列表 获取旅游景点列表 获取景点产品列表 存储数据 1 获取出发地站点 (1)访问touch.qunar.com (2)按F12,单击自由行,在自由行页面点击搜索框 (3 ...

  3. liunx mysql数据库目录迁移

    1.查看mysql安装目录 从目录etc/my.cnf中查看安装目录 2.进入mysql目录,停止mysql服务 命令: cd usr/local/mysql 命令:service mysql sto ...

  4. FZU2018级算法第一次作业 1.1fibonacci (矩阵快速幂)

    题目 Winder最近在学习fibonacci 数列的相关知识.我们都知道fibonacci数列的递推公式是F(n)=F(n-1)+F(n-2)(n>=2 且n 为整数). Winder想知道的 ...

  5. Linux环境下错误码及意义总结

    Linux的错误码包含在/usr/include/asm-generic/errno-base.h和/usr/include/asm-generic/errno.h 这两个文件内: #ifndef _ ...

  6. go 连接到数据库

    package main import ( "fmt" _ "github.com/go-sql-driver/mysql" "github.com/ ...

  7. nginx安装错误:No package nginx available

    出现错误: 1,备份 mv /etc/yum.repos.d/CentOS-Base.repo /etc/yum.repos.d/CentOS-Base.repo.backup 2.下载新的CentO ...

  8. "CreateProcess error=206, 文件名或扩展名太长。",用gradle构建项目创建mapper文件时提示这个错误,是Windows Gradle长类路径问题,官方已经修复

    用gradle构建项目mapper文件时,提示这个错误,这个是Windows Gradle长类路径问题, gradle官方已经解决了这个问题. 官网给出的解决方法地址:https://plugins. ...

  9. [高清·非影印]Spring实战+SpringBoot实战+Spring微服务实战+SpringCloud微服务实战(全4本)

    ------ 郑重声明 --------- 资源来自网络,纯粹共享交流, 如果喜欢,请您务必支持正版!! --------------------------------------------- 下 ...

  10. 论文笔记 Large Pose 3D Face Reconstruction from a Single Image via Direct Volumetric CNN Regression

    Large Pose 3D Face Reconstruction from a Single Image via Direct Volumetric CNN Regression 该文献采用一个新型 ...