传送

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

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

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

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

枚举完行,列,就计算一次当前子矩阵的分数,与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. JS和C#后台获取网站URL

    例:网页URL :  http://localhost:8086/index.aspx?topicId=361 1.设置或获取 href 属性中跟在问号后面的部分:window.location.se ...

  2. github javascript相关项目star数排行榜(前30,截止2016.11.18):

    github javascript相关项目star数排行榜(前30,截止2016.11.18): 前端开源框架 TOP 100 前端 TOP 100:::::https://www.awesomes. ...

  3. docker--docker架构

    4 docker 架构 Docker uses a client-server architecture. The Docker client talks to the Docker daemon, ...

  4. python进阶之类的反射

    有应用场景的技术才是有灵魂的技术------>最近同时问我,在python中,给你一个函数或者类的字符串名称,你怎么得到该函数和类,以下结合源码记录我得到的方式: 1.给一个函数的字符串&quo ...

  5. 1/n循环节长度

    /* * 求1/i的循环节长度的最大值,i<=n */ ; int res[MAXN]; // 循环节长度 int main() { memset(res, , sizeof(res)); in ...

  6. C# ASP.NET 手写板并生成图片保存

    前端: @{ Layout = null; } <!DOCTYPE html> <html lang="zh-CN"> <head> <t ...

  7. mybatis复习笔记(1):

    一.简介:什么是MyBatis 1.MyBatis是一款优秀的持久层框架,支持定制化SQL.存储过程以及高级映射.MyBatis避免了几乎所有的JDBC代码和手动设置参数以及获取结果集.MyBatis ...

  8. python学习笔记(7)文件的访问与函数式编程

    一.文件读写的3中方法 1.直接读入 fiel1=open('test.txt') file2=open('output.txt') while True: line=file1.readLine() ...

  9. webGL 旋转算法

    lon = 0,//y旋转角度 lat = 0, onMouseDownLat = 0, phi = 0, theta = 0; if ( isUserInteracting === false ) ...

  10. nginx中break和last的区别

    一.环境准备 资源文件创建 mkdir -p /opt/tmp/wqy/test/aaa mkdir -p /opt/tmp/wqy/test/bbb echo "aaa" > ...