传送

一道看起来就很暴力的题。

这道题不仅暴力,还要用正确的姿势打开暴力。

因为子矩阵的参数有两个,一个行一个列(废话)

我们一次枚举两个参数很容易乱对不对?所以我们先枚举行,再枚举列

枚举完行,列,就计算一次当前子矩阵的分数,与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子矩阵的更多相关文章

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

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

  2. 洛谷P2258 子矩阵

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

  3. P2258 子矩阵(dp)

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

  4. P2258 子矩阵——搜索+dp

    P2258 子矩阵 二进制枚举套二进制枚举能过多一半的点: 我们只需要优化一下第二个二进制枚举的部分: 首先我们先枚举选哪几行,再预处理我们需要的差值,上下,左右: sum_shang,sum_hen ...

  5. P2258 子矩阵

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

  6. 洛谷 P2258 子矩阵

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

  7. luogu P2258 子矩阵 |动态规划

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

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

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

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

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

随机推荐

  1. 详解微信小程序支付流程

    转发博主 https://blog.csdn.net/qq_38378384/article/details/80882980 花了几天把小程序的支付模块接口写了一下,可能有着公众号开发的一点经验,没 ...

  2. Vue --》 如何在vue中调用百度地图

    1.项目根目录下下载百度地图插件 npm install vue-baidu-map –save 2.在首页index.html中引入百度地图: <script type="text/ ...

  3. PTA第四周作业

    一.本周完成的作业 7-2 选择法排序 (20 分) 本题要求将给定的n个整数从大到小排序后输出. 输入格式: 输入第一行给出一个不超过10的正整数n.第二行给出n个整数,其间以空格分隔. 输出格式: ...

  4. 2019寒假作业一:PTA7-1 打印沙漏

    - 打印沙漏 ( 分) 本题要求你写个程序把给定的符号打印成沙漏的形状.例如给定17个“*”,要求按下列格式打印 ***** *** * *** ***** 所谓“沙漏形状”,是指每行输出奇数个符号: ...

  5. Java实验报告(一)&&第三周学习总结

    实验报告(一) 1. 打印输出所有的“水仙花数”,所谓“水仙花数”是指一个3位数,其中各位数字立方和等于该数本身.例如,153是一个“水仙花数”. 源代码: public class Main { p ...

  6. 编码规范(code style guide)

    1. Javascript Google: https://google.github.io/styleguide/jsguide.html Airbnb:https://github.com/air ...

  7. CentOS7 修复MBR引导

    为了达到实验目的,首先破坏MBR引导bootloader 重启系统发现系统进不去了,这正是我们想要的 重启进入系统救援模式,输入以下命令重建MBR引导bootloader 重启,可以正常引导进入系统

  8. Java数据结构介绍(线性结构和非线性结构)

    数据结构包括:线性结构和非线性结构. 线性结构 数据元素之间存在一对一的线性关系 包括顺序存储结构和链式存储结构.顺序存储的线性表称为顺序表,顺序表中的存储元素是连续的 链式存储的线性表称为链表,链表 ...

  9. Spring Boot 集成 Ehcache 缓存,三步搞定!

    作者:谭朝红 www.ramostear.com/articles/spring_boot_ehcache.html 本次内容主要介绍基于Ehcache 3.0来快速实现Spring Boot应用程序 ...

  10. k8s结合helm部署

    一.安装Helm helm教程以及安装可以参考这篇文章 二.Heml说明 常见的helm模板如下 myapp - chart 包目录名 ├── charts - 依赖的子包目录,里面可以包含多个依赖的 ...