[bzoj1084][SCOI2005]最大子矩阵_动态规划_伪·轮廓线dp

最大子矩阵 bzoj-1084 SCOI-2005

题目大意:给定一个n*m的矩阵,请你选出k个互不重叠的子矩阵使得它们的权值和最大。

注释:$1\le n \le 100$,$1\le m\le 2$,$1\le k\le 10$。

想法:不会。。。看了数据范围..卧槽?m<=2?????我们就可以进行一个简单的轮廓线dp。

首先,先分m==1和m==2分类讨论,m==1不说了

m==2

令f[k][i][j]是第一列到了i,第二列到了j,已经选取了k个矩形的最大权值。

转移:有3种转移方式:

1.从左侧转移:f[k][i][j]=max(f[k][i][j] , f[k-1][l][j] + before[1][i] - before[1][l] )。

2.从右侧转移,转移方程同理。

3.选取横跨左右的矩阵,此时必须有i==j:f[k][i][j]=max(f[k][i][j] , f[k-1][l][l] + before[1][i] + before[2][i] - before[1][l] - before[2][l])。

时间复杂度:$O(n^3\cdot k)$。

最后,附上丑陋的代码... ...

#include<iostream>
#include<cstdio>
#define N 110
using namespace std;
int F[12][N],f[12][N][N];
int n,m,K,s[3][N];
int a;
void dispose1()
{
    for(int k=1;k<=K;k++)
        for(int i=1;i<=n;i++)
        {
            F[k][i]=F[k][i-1];
            for(int j=0;j<=i-1;j++) F[k][i]=max(F[k][i],F[k-1][j]+s[1][i]-s[1][j]);
        }
    int ans=0;
    for(int i=0;i<=K;i++) ans=max(ans,F[i][n]);
    printf("%d\n",ans);
}
void dispose2()
{
    for(int i=1;i<=n;i++) s[0][i]=s[1][i]+s[2][i];
    for(int k=1;k<=K;k++) for(int i=1;i<=n;i++)
        for(int j=1;j<=n;j++)
        {
            f[k][i][j]=max(f[k][i-1][j],f[k][i][j-1]);
            for(int l=0;l<i;l++) f[k][i][j]=max(f[k][i][j],f[k-1][l][j]+s[1][i]-s[1][l]);
            for(int l=0;l<j;l++) f[k][i][j]=max(f[k][i][j],f[k-1][i][l]+s[2][j]-s[2][l]);
            if(i==j)  for(int l=0;l<i;l++) f[k][i][j]=max(f[k][i][j],f[k-1][l][l]+s[0][i]-s[0][l]);
        }
    int ans=0;
    for(int i=0;i<=K;i++) ans=max(ans,f[i][n][n]);
    printf("%d\n",ans);
}
int main()
{
    scanf("%d%d%d",&n,&m,&K);
    for(int i=1;i<=n;i++)
        for(int j=1;j<=m;j++)
        {
            scanf("%d",&a);
            s[j][i]=a+s[j][i-1];
        }
    if(m==1) dispose1();
    else dispose2();
    return 0;
}

  小结:一般情况下,我们能求解的只是简单问题的全部版本和复杂问题的特殊版本,所以要对问题抱有敬畏之心(Orz石总)

原文地址:https://www.cnblogs.com/ShuraK/p/9265001.html

时间: 2024-08-10 06:37:27

[bzoj1084][SCOI2005]最大子矩阵_动态规划_伪·轮廓线dp的相关文章

BZOJ1084 [SCOI2005]最大子矩阵

Description 这里有一个n*m的矩阵,请你选出其中k个子矩阵,使得这个k个子矩阵分值之和最大.注意:选出的k个子矩阵不能相互重叠. Input 第一行为n,m,k(1≤n≤100,1≤m≤2,1≤k≤10),接下来n行描述矩阵每行中的每个元素的分值(每个元素的分值的绝对值不超过32767). Output 只有一行为k个子矩阵分值之和最大为多少. Sample Input 3 2 2 1 -3 2 3 -2 3 Sample Output 9 正解:DP 解题报告: 乍一看,感觉像是才

算法导论_动态规划_最长公共子序列

一.动态规划的概念 动态规划(Dynamic Programming)是通过组合子问题的解而解决整个问题的.分治是指将问题划分成一些独立的子问题,递归地求解各子问题,然后合并子问题的解而得到原始问题的解,与此不同,动态规划适用于子问题不是独立的情况,也就是各个子问题包含公共的子问题.在这种情况下,采用分治法会做许多不必要的工作,即重复地求解公共地子问题.动态规划算法对每个子问题只求解一次,将其结果保存在一张表中,从而避免每次遇到各个子问题时重新计算答案. 动态规划通常应用于最优化问题.此类问题可

算法导论_动态规划_最长回文子序列

一.问题的描述 回文序列(Palindromic sequence, Palindrome)是指正向遍历和反向遍历完全相同的序列,例如字符串"AAAAA"显然是一个回文序列,又如字符串"[email protected]"也是一个回文序列.现在,我们要在一个(字符)序列中找出最长回文子序列的长度.例如字符序列"BBABCBCAB",最长回文子序列是"BACBCAB"(可能不唯一),它的长度是7:子序列"BBBBB&q

[bzoj2748][HAOI2012]音量调节_动态规划_背包dp

音量调节 bzoj-2748 HAOI-2012 题目大意:有一个初值,给你n个$\delta$值,求最后不超过给定的限制的情况下的改变的最大值.每个$\delta$值可以+也可以-. 注释:$1\le n\le 50$,$1\le 限制\le 1000$. 想法:正负背包.在背包的之后更新用两行更新,保证每一个$\delta$值都被计入了答案. 最后,附上丑陋的代码... ... #include <iostream> #include <cstdio> #include <

[bzoj1879][Sdoi2009]Bill的挑战_动态规划_状压dp

Bill的挑战 bzoj-1879 Sdoi-2009 题目大意: 注释:$1\le t \le 5$,$1\le m \le 15$,$1\le length \le 50$. 想法: 又是一个看数据范围想做法的题,我们想到状压dp. 看了题解... ...网上给的状态是f[len][s]表示长度为len满足状态s的字符串个数. 光看状态... ...可能算重啊?! 其实... ... 状态:dp[len][s]表示长度为len,能且只能满足状态为s的字符串个数. 转移:我们先预处理出g[i]

[bzoj4282]慎二的随机数列_动态规划_贪心

慎二的随机数列 bzoj-4282 题目大意:一个序列,序列上有一些数是给定的,而有一些位置上的数可以任意选择.问最长上升子序列. 注释:$1\le n\le 10^5$. 想法:结论:逢N必选.N是可以任意选择的位置. 具体的,我们将所有N踢出序列,将给定的权值-=前面N的个数.再在当前序列上求最长上升子序列. 正确性的话如果当前序列中的数: 如果前面的数小于后面的数,显然中间的N我也可以加上. 如果前面的数大于后面的数: 如果前面的数在原序列中的权值大于后面的数在原序列中的权值,那么这两个数

[bzoj3622]已经没有什么好害怕的了_动态规划_容斥原理

bzoj-3622 已经没有什么好害怕的了 题目大意: 数据范围:$1\le n \le 2000$ , $0\le k\le n$. 想法: 首先,不难求出药片比糖果小的组数. 紧接着,我开始的想法是 $f_{(i,j)}$表示前$i$个糖果中,满足糖果比药片大的组数是$j$的方案数. 进而发现需要将两个数组排序. 到这里一切都很正常,但是我们发现了一个问题:就是我在转移的时候,分两种情况讨论.第一种是当前糖果配对的药片比自己大,第二种是比自己小. 这样的话我需要乘上两个组合数. 但是我们仔细

全职太太投资P2P 能否赚到零花钱 P2P 投资 零花钱_新浪财经_新浪网

市民张女士:最近看了好多P2P理财,有不少人赚了钱的,也有好多跑路血本无归的.现在孩子上小学一年级了,我在家没啥事,创业做小生意老公不同意,搞搞理财投资他还是赞成的.我们积蓄不多,房子供完了,儿子上学了,就天天琢磨网上P2P平台.有的名声很大,但收益不高,就比银行存款多1-2%的利息:有的名气不大,但收益有20%左右.看了手痒,又怕投资被跑路了.老公给的零花钱每次给得不多,我不舍得花攒起来了,想自己投投资,理理财,边带儿子边研究.现在的问题是,这种投资风险有哪些?会不会涉嫌非法集资?怎么判断哪些

1084: [SCOI2005]最大子矩阵

1084: [SCOI2005]最大子矩阵 Description 这里有一个n*m的矩阵,请你选出其中k个子矩阵,使得这个k个子矩阵分值之和最大.注意:选出的k个子矩阵不能相互重叠. Input 第一行为n,m,k(1≤n≤100,1≤m≤2,1≤k≤10),接下来n行描述矩阵每行中的每个元素的分值(每个元素的分值的绝对值不超过32767). Output 只有一行为k个子矩阵分值之和最大为多少. Sample Input 3 2 2 1 -3 2 3 -2 3 Sample Output 9