• Luogu 3625 [APIO2009]采油区域


    想了很久的dp,看了一眼题解之后感觉自己被安排了。

    发现从一个矩形中选择三个不相交的正方形一共只有六种取法。

    那么我们可以处理出四个值:

    $f_{i, j}$分别表示以$(i, j)$为右下角,左下角,右上角,左上角的矩阵中选一个$k*k$正方形的最大值。

    这样就可以算出前四种情况,后两种情况只要乱搞就可以了。

    时间复杂度$O(nm)$。

    Code:

    #include <cstdio>
    #include <cstring>
    using namespace std;
    
    const int N = 1505;
    
    int n, m, k, ans, a[N][N], sum[N][N], s[N][N];
    int f1[N][N], f2[N][N], f3[N][N], f4[N][N], r[N], c[N];
    
    template <typename T>
    inline void read(T &X) {
        X = 0; char ch = 0; T op = 1;
        for(; ch > '9'|| ch < '0'; ch = getchar())
            if(ch == '-') op = -1;
        for(; ch >= '0' && ch <= '9'; ch = getchar())
            X = (X << 3) + (X << 1) + ch - 48;
        X *= op;
    }
    
    inline void chkMax(int &x, int y) {
        if(y > x) x = y;
    }
    
    inline int max(int x, int y) {
        return x > y ? x : y;
    }
    
    inline int max(int x, int y, int z) {
        return max(max(x, y), z);
    }
    
    int main() {
        read(n), read(m), read(k);
        for(int i = 1; i <= n; i++)
            for(int j = 1; j <= m; j++) {
                read(a[i][j]);
                sum[i][j] = sum[i][j - 1] + a[i][j];
            }
        for(int i = 1; i <= n; i++)
            for(int j = 1; j <= m; j++)
                sum[i][j] += sum[i - 1][j];
        for(int i = k; i <= n; i++)
            for(int j = k; j <= m; j++)
                s[i][j] = sum[i][j] + sum[i - k][j - k] - sum[i][j - k] - sum[i - k][j];
                
    /*    for(int i = 1; i <= n; i++, printf("
    "))
            for(int j = 1; j <= m; j++)
                printf("%d ", s[i][j]);      */
    
        for(int i = 1; i <= n; i++)
            for(int j = 1; j <= m; j++)
                chkMax(f1[i][j], max(s[i][j], f1[i - 1][j], f1[i][j - 1]));
        for(int i = 1; i <= n; i++)
            for(int j = m; j >= 1; j--)
                chkMax(f2[i][j], max(s[i][j + k - 1], f2[i - 1][j], f2[i][j + 1]));
        for(int i = n; i >= 1; i--)
            for(int j = m; j >= 1; j--)
                chkMax(f3[i][j], max(s[i + k - 1][j + k - 1], f3[i + 1][j], f3[i][j + 1]));
        for(int i = n; i >= 1; i--)
            for(int j = 1; j <= m; j++)
                chkMax(f4[i][j], max(s[i + k - 1][j], f4[i + 1][j], f4[i][j - 1])); 
        
        ans = 0;
        for(int i = k; i <= n - k; i++)
            for(int j = k; j <= m - k; j++) {
                chkMax(ans, f1[i][j] + f2[i][j + 1] + f3[i + 1][1]);
                chkMax(ans, f1[i][j] + f2[n][j + 1] + f4[i + 1][j]);
                chkMax(ans, f1[n][j] + f2[i][j + 1] + f3[i + 1][j + 1]);
                chkMax(ans, f1[i][m] + f4[i + 1][j] + f3[i + 1][j + 1]);
            }
        
        for(int i = 1; i <= n; i++)
            for(int j = 1; j <= m; j++) {
                chkMax(r[i], s[i][j]);
                chkMax(c[j], s[i][j]);
            }
        
        for(int i = k; i <= n - 2 * k; i++)
            for(int j = i + k, mid = r[j]; j <= n - k; j++, chkMax(mid, r[j]))
                chkMax(ans, f1[i][m] + mid + f3[j + 1][1]);
        for(int i = k; i <= m - 2 * k; i++)
            for(int j = i + k, mid = c[j]; j <= m - k; j++, chkMax(mid, c[j]))
                chkMax(ans, f1[n][i] + mid + f2[n][j + 1]);
        
        printf("%d
    ", ans);
        return 0;
    }
    View Code
  • 相关阅读:
    java中的数组长度的计算
    用两个栈来实现一个队列,完成队列的Push和Pop操作。 队列中的元素为int类型。C++实现
    c++中计算数组的长度。以及c++中向量的长度的计算的方式。
    3.mouseenter和mouseover事件的区别
    0.jQuery选择器
    2.点击隐藏盒子
    1.jQuery入口函数
    jquery选项卡效果
    %你考试2020.1
    二十七、rsync同步工具
  • 原文地址:https://www.cnblogs.com/CzxingcHen/p/9562563.html
Copyright © 2020-2023  润新知