P2258子矩阵





一道看起来就很暴力的题。
这道题不仅暴力,还要用正确的姿势打开暴力。
因为子矩阵的参数有两个,一个行一个列(废话)
我们一次枚举两个参数很容易乱对不对?所以我们先枚举行,再枚举列
枚举完行,列,就计算一次当前子矩阵的分数,与ans取min。
代码:
但是复杂度会高到爆炸。来我们深吸一口O2。
// luogu-judger-enable-o2
#include<bits/stdc++.h>
using namespace std;
int ma[][],n,m,r,c,ans=,ch[],cl[];
void js()
{ int an=;
for(int i=;i<=r;i++)//求子矩阵左右的差的绝对值
for(int j=;j<=c;j++)
an+=abs(ma[ch[i]][cl[j]]-ma[ch[i]][cl[j-]]);
for(int i=;i<=r;i++)//求上下的差的绝对值
for(int j=;j<=c;j++)
an+=abs(ma[ch[i]][cl[j]]-ma[ch[i-]][cl[j]]);
ans=min(ans,an);
}
void dfs(int x,int y,int nr,int nc)//x:当前枚举的行在原矩阵中是第x行,y:当前枚举的列在原矩阵中是第y列,nr:将要枚举的行的数量,nc:将要枚举的列的数量
{
if(nc==c+)
{
js();
return;
}
if((x>n&&nr!=r+)||(y>m&&nc!=c+))return ;
if(nr==r+)//当枚举完行了之后,枚举列
{
for(int i=y;i<=m;i++)
{
cl[nc]=i;
dfs(x,i+,nr,nc+);
}
}
else//枚举行
{
for(int i=x;i<=n;i++)
{
ch[nr]=i;
dfs(i+,y,nr+,nc);
}
}
}
int main()
{
scanf("%d%d%d%d",&n,&m,&r,&c);
for(int i=;i<=n;i++)
for(int j=;j<=m;j++)
scanf("%d",&ma[i][j]);
dfs(,,,);
printf("%d",ans);
}
TTTLE
还是T成这种美丽的图形

你看它T的多有美感
好了我们仔(ting)细(xue)思(zhang)考(jiang)怎么不吸氧通过剩下这几个点
能(xue)想(zhang)到(shuo)用dp来解决这道题。
但是这道题实在是太暴力了,所以我们还是得先暴力一下。
我们依旧先枚举行。然后对列进行dp。
设ver[i]为第i列的元素上下之差的绝对值之和,del[i][j]是第i列与第j列每行的元素之差的绝对值之和,d[i][j]为前i列,选择了j列,这j列中一定有第i列时的最小得分。
简单的画一下ver和del
ver:

每一个元素减上面那个元素的差的绝对值加起来就是ver
del:

每一行右边减左边的差的绝对值加起来就是del
我们考虑dp[i][j]的转移方程
dp[i][j]可以由dp[i-1][j-1]再选上第i列得到,可以由dp[i-2][j-1]再选第i列得到,可以由dp[i-3][j-1]得到……
所以状态转移方程就是dp[i][j]=min{dp[i][j],dp[i-k][j-1]+ver[i]+del[i-k][i]}(1<=k,i-k>=j-1)
最终答案就是dp[i][c](c<=i<=m)的最小值
#include<bits/stdc++.h>
using namespace std;
int ma[][],n,m,r,c,ans=,ver[],del[][],d[][],ch[];//ch是选择的行
void dp()
{
memset(d,,sizeof(d));//千万不要memset成0
memset(ver,,sizeof(ver));
memset(del,,sizeof(del));
for(int i=;i<=m;i++)//枚举第i列
for(int j=;j<=r;j++)//枚举行的编号
ver[i]+=abs(ma[ch[j]][i]-ma[ch[j-]][i]);
for(int i=;i<=m;i++)//第i列
for(int j=i+;j<=m;j++)//第j列
for(int k=;k<=r;k++)//枚举行(注意循环的顺序)
del[i][j]+=abs(ma[ch[k]][j]-ma[ch[k]][i]);
for(int i=;i<=m;i++)
d[i][]=ver[i];
for(int i=;i<=m;i++)
for(int j=;j<=c;j++)
for(int k=;k<i&&i-k>=j-;k++)
d[i][j]=min(d[i][j],d[i-k][j-]+ver[i]+del[i-k][i]); for(int i=c;i<=m;i++)
ans=min(ans,d[i][c]);
}
void dfs(int x,int nr)//x,nr的含义与爆搜的含义相同
{ if(nr==r+)
{
dp();
return;
}
if(x>n)return ;
for(int i=x;i<=n;i++)
{
ch[nr]=i;
dfs(i+,nr+);
}
}
int main()
{
scanf("%d%d%d%d",&n,&m,&r,&c);
for(int i=;i<=n;i++)
for(int j=;j<=m;j++)
scanf("%d",&ma[i][j]);
dfs(,);
printf("%d",ans);
}
P2258子矩阵的更多相关文章
- 洛谷 P2258 子矩阵 解题报告
P2258 子矩阵 题目描述 给出如下定义: 子矩阵:从一个矩阵当中选取某些行和某些列交叉位置所组成的新矩阵(保持行与列的相对顺序)被称为原矩阵的一个子矩阵. 例如,下面左图中选取第 2 . 4行和第 ...
- 洛谷P2258 子矩阵
P2258 子矩阵 题目描述 给出如下定义: 子矩阵:从一个矩阵当中选取某些行和某些列交叉位置所组成的新矩阵(保持行与列的相对顺序)被称为原矩阵的一个子矩阵. 例如,下面左图中选取第2.4行和第2.4 ...
- P2258 子矩阵(dp)
P2258 子矩阵 题目描述 给出如下定义: 子矩阵:从一个矩阵当中选取某些行和某些列交叉位置所组成的新矩阵(保持行与列的相对顺序)被称为原矩阵的一个子矩阵. 例如,下面左图中选取第2.4行和第2.4 ...
- P2258 子矩阵——搜索+dp
P2258 子矩阵 二进制枚举套二进制枚举能过多一半的点: 我们只需要优化一下第二个二进制枚举的部分: 首先我们先枚举选哪几行,再预处理我们需要的差值,上下,左右: sum_shang,sum_hen ...
- P2258 子矩阵
题目描述 给出如下定义: 子矩阵:从一个矩阵当中选取某些行和某些列交叉位置所组成的新矩阵(保持行与列的相对顺序)被称为原矩阵的一个子矩阵. 例如,下面左图中选取第 222 . 444 行和第 222 ...
- 洛谷 P2258 子矩阵
题目描述 给出如下定义: 子矩阵:从一个矩阵当中选取某些行和某些列交叉位置所组成的新矩阵(保持行与列的相对顺序)被称为原矩阵的一个子矩阵. 例如,下面左图中选取第2.4行和第2.4.5列交叉位置的元素 ...
- luogu P2258 子矩阵 |动态规划
题目描述 给出如下定义: 子矩阵:从一个矩阵当中选取某些行和某些列交叉位置所组成的新矩阵(保持行与列的相对顺序)被称为原矩阵的一个子矩阵. 例如,下面左图中选取第22.44行和第22.44.55列交叉 ...
- 洛谷P2258 子矩阵[2017年5月计划 清北学堂51精英班Day1]
题目描述 给出如下定义: 子矩阵:从一个矩阵当中选取某些行和某些列交叉位置所组成的新矩阵(保持行与列的相对顺序)被称为原矩阵的一个子矩阵. 例如,下面左图中选取第2.4行和第2.4.5列交叉位置的元素 ...
- 洛谷P2258 子矩阵 题解 状态压缩/枚举/动态规划
作者:zifeiy 标签:状态压缩.枚举.动态规划 题目链接:https://www.luogu.org/problem/P2258 这道题目状态压缩是肯定的,我们需要用二进制来枚举状态. 江湖上有一 ...
随机推荐
- Win10.去掉任务栏缩略图(just4explorer)
ZC: 该方式只适用于 Explorer(即 WIndows任务管理器),Why? ∵ 看文中设置 regedit 的路径,它设置的就是 Explorer下的键值 ... 1.HKEY_CURRENT ...
- Tensorflow之CNN卷积层池化层padding规则
padding的规则 · padding=‘VALID’时,输出的宽度和高度的计算公式(下图gif为例) 输出宽度:output_width = (in_width-filter_wid ...
- [Git] 013 远程仓库篇 第零话 使用前的一些配置
0. 前言 本地仓库和 GitHub 上的远程仓库之间的传输是通过 "SSH" 加密的,所以使用前需要进行一些设置 这回的任务 设置"身份象征" 创建 &quo ...
- Sentinel限流实现原理
Sentinel限流的神秘面纱: 之前我们学习过限流比较主流的三种算法:漏桶,令牌桶,滑动窗口.而Sentinel采用的是最后一种,滑动窗口来实现限流的. 通过对Sentinel基础Api的使用,我们 ...
- Java-集合第三篇List集合
1.List集合 有序可重复集合,集合中的每个元素都有其对应的顺序索引. 2.List相对于Collection额外提供的方法: 1>void add(int index,Object elem ...
- enWin使用部分中文字库
在小型嵌入式设备中有时需要简单的人机2交互界面,小型GUI有很多,比较常用的有STenWin,UCGUI,enwin,Embedded Wizard GUI.对与STenWin和enWin区别主要在S ...
- void die(const char *msg)
void die(const char *msg) { perror(msg); exit(errno); }
- P5444 [APIO2019]奇怪装置
传送门 考虑求出最小的循环节 $G$ 使得 $t,t+G$ 得到的数对是一样的 由 $y \equiv t \mod B$ ,得到 $G$ 一定是 $B$ 的倍数,设 $zB=G$,则 $t,t+zB ...
- asp.net Swiper 轮播动画
原文:https://blog.csdn.net/qq_39656138/article/details/90451289 官网:https://www.swiper.com.cn/api/index ...
- SSH简单概念
Spring:轻量级控制反转(IoC)和面向切面(AOP)的容器框架,让对象与对象之间的关系通过配置文件来管理,减低耦合度 IoC:凡是在容器中配置过的对象才会有Spring提供的服务和功能 AOP: ...