传送

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

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

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

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

枚举完行,列,就计算一次当前子矩阵的分数,与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. debian下使用shell脚本时出现了 declare:not found 解决方法

    问题:出现declare:not found的提示 解决:原来,UBUNTU用的是dash(后来证明这个其实这个不是错误的原因:从#!/bin/bash到#!/bin/dash,依旧无法运行,在这写出 ...

  2. 20190827 On Java8 第十四章 流式编程

    第十四章 流式编程 流的一个核心好处是,它使得程序更加短小并且更易理解.当 Lambda 表达式和方法引用(method references)和流一起使用的时候会让人感觉自成一体.流使得 Java ...

  3. js模块化编程之CommonJS和AMD/CMD!

    有了模块,我们就可以更方便地使用别人的代码,想要什么功能,就加载什么模块. 但是,这样做有一个前提,那就是大家必须以同样的方式编写模块,否则你有你的写法,我有我的写法,岂不是乱了套! 于是下面三个模块 ...

  4. IIS中配置Office COM组件 [dufu图解系列]

    解决问题: 读写office文档目前有2中方法,一种为引用微软的office com组件(下图为Microsoft Office 16.0 Object Library), 另一种为用第三方组件NPO ...

  5. 【Python—字典的用法】创建字典的3种方法

    #创建一个空字典 empty_dict = dict() print(empty_dict) #用**kwargs可变参数传入关键字创建字典 a = dict(one=1,two=2,three=3) ...

  6. Redis--小小总结

    1.基本定义 memcached是纯粹的key-value内存数据库,也可能不应该叫数据库,应该叫另类缓存技术: Redis是一个基于内存的高性能key-value数据库:将数据全部加载到内存中,并定 ...

  7. Java学习day4 程序流程控制一

    一.分支结构 条件语句:if...else if语句: 一个 if 语句包含一个布尔表达式和一条或多条语句,如果布尔表达式的值为 true,则执行 if 语句中的代码块,否则执行 if 语句块后面的代 ...

  8. python 学习第四十七天shelve模块

    shelve模块是一个简单的k,v将内存数据通过文件持久化的模块,可以持久化任何pickle可支持的python数据格式. 1,序列化 import shelve f=shelve.open('she ...

  9. 看漫画就能学SQL,简直太cool了

    对于SQl, 很多人学不会的原因是从一开始就没明白,学这东西能干啥,学会了能有什么用.甚至有些人不知道'SQL'应该怎么读,以至于一开始兴致勃勃,但是学到一半放弃了. 注意:'sql'真的不能读成'烧 ...

  10. C# asp.net XML格式的字符串显示不全

    前台显示XML字符串显示不全 后台XML字符串使用<xmp></xmp>将XML格式字符串括起来