@topcoder - SRM611D1L3@ ElephantDrinking


@description@

给定有一个 n*n 的平原,其中一些格子有些泉水。泉水每单位时间生产的水量有多有少。

平原周围有 4*n 头大象,如下图所示的绿色部分。大象可以将鼻子笔直伸向自己面对的方向喝泉水(假设鼻子长到可以触碰对面边缘),如下图所示红色线段。

本题还有些额外限制:大象的鼻子不能相交;每个泉水最多被一个大象占领。如图 (a) 是合法的,图 (b) 都算大象的鼻子相交。
请计算大象们每单位时间最多能喝多少的水。

Class:
ElephantDrinking
Method:
maxSum
Parameters:
String[]
Returns:
int

Constraints
n 在 2 到 50 之间。通过字符串数组来描述平原,其中字符串 i 的第 j 位为一个 '0'~'9' 的数,描述平原 (i, j) 的单位时间产水量(0 则不产水)。

Examples
0)
{"00000",
"00110",
"01000",
"00100",
"00000"}
Returns: 4
1)
{"111",
"191",
"111"}
Returns: 16

@solution@

这个 dp。。。虽然不难理解。。。但是好像有点乱搞。。。
(注:以下的图片都不是我画的,是从其他地方截下来的)

首先我们令 a 表示最左面的大象的鼻子向右最远延伸到多少行,同理定义 b 表示最右,c 表示最下,d 表示最上。
考虑如果 a <= b,实际上中间形成了一条可以供上下的大象自由伸展鼻子的区域(如下图)。

我们假设预处理出 f[i][j] 表示仅考虑左大象和上大象时,取矩阵左上部分到 (i, j) 这一区域的最优答案。同理再预处理出仅考虑左、下;仅考虑右、上;仅考虑右、下的值。
则上图中蓝、黄、青、紫区域的答案就是我们预处理出来的值。
橙色区域实际上每一列只会有两头大象,所以处理出每一列的最大值与次大值之和即可。

如果 d <= c,其实就是把上下变成了左右,一样的处理即可。

如果 a > b 且 d > c,实际上形成了像是弦图的模样(如下图):

只需要枚举中间那个空的矩形即可,一样利用我们预处理好的信息。
当然它还可以转个方向(如下图),不过处理方法还是一样:

现在考虑怎么预处理出 f[i][j] 呢?这里仅说明左、上的情况,其他情况类似。
我们枚举最后一行的大象鼻子延伸的位置;再枚举一列,使那一列上面的大象鼻子延伸至最后一行,且那一列左边没有任何大象延伸至最后一行。枚举的那一列右边的大象没有限制,故肯定延伸至最大泉水处。

这样总时间复杂度就是 O(n^4) 的,可以很轻松地跑过。

@accepted code@

#include<cstdio>
#include<vector>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
const int MAXN = 50;
class ElephantDrinking{
    public:
    int a[MAXN + 5][MAXN + 5], n;
    int tmp[MAXN + 5], res;
    int f1[MAXN + 5][MAXN + 5];
    void get_lu() {
        for(int j=1;j<=n;j++) tmp[j] = 0;
        memset(f1, 0, sizeof f1);
        for(int i=1;i<=n;i++) {
            for(int j=1;j<=n;j++) {
                tmp[j] = max(tmp[j], a[i][j]);
                for(int k=0;k<=j;k++) {
                    res = a[i][k], f1[i][j] = max(f1[i][j], res + f1[i-1][j]);
                    for(int l=j;l>k;l--)
                        res += tmp[l], f1[i][j] = max(f1[i][j], res + f1[i-1][l-1]);
                }
//              printf("* %d %d : %d\n", i, j, f1[i][j]);
            }
        }
    }
    int f2[MAXN + 5][MAXN + 5];
    void get_ld() {
        for(int j=1;j<=n;j++) tmp[j] = 0;
        memset(f2, 0, sizeof f2);
        for(int i=n;i>=1;i--) {
            for(int j=1;j<=n;j++) {
                tmp[j] = max(tmp[j], a[i][j]);
                for(int k=0;k<=j;k++) {
                    res = a[i][k], f2[i][j] = max(f2[i][j], res + f2[i+1][j]);
                    for(int l=j;l>k;l--)
                        res += tmp[l], f2[i][j] = max(f2[i][j], res + f2[i+1][l-1]);
                }
//              printf(". %d %d : %d\n", i, j, f2[i][j]);
            }
        }
    }
    int f3[MAXN + 5][MAXN + 5];
    void get_ru() {
        for(int j=1;j<=n;j++) tmp[j] = 0;
        memset(f3, 0, sizeof f3);
        for(int i=1;i<=n;i++) {
            for(int j=n;j>=1;j--) {
                tmp[j] = max(tmp[j], a[i][j]);
                for(int k=n+1;k>=j;k--) {
                    res = a[i][k], f3[i][j] = max(f3[i][j], res + f3[i-1][j]);
                    for(int l=j;l<k;l++)
                        res += tmp[l], f3[i][j] = max(f3[i][j], res + f3[i-1][l+1]);
                }
//              printf("? %d %d : %d\n", i, j, f3[i][j]);
            }
        }
    }
    int f4[MAXN + 5][MAXN + 5];
    void get_rd() {
        for(int j=1;j<=n;j++) tmp[j] = 0;
        memset(f4, 0, sizeof f4);
        for(int i=n;i>=1;i--) {
            for(int j=n;j>=1;j--) {
                tmp[j] = max(tmp[j], a[i][j]);
                for(int k=n+1;k>=j;k--) {
                    res = a[i][k], f4[i][j] = max(f4[i][j], res + f4[i+1][j]);
                    for(int l=j;l<k;l++)
                        res += tmp[l], f4[i][j] = max(f4[i][j], res + f4[i+1][l+1]);
                }
//              printf("! %d %d : %d\n", i, j, f4[i][j]);
            }
        }
    }
    int solve1() {
        int ret = 0, res1 = 0, res2 = 0, res3 = 0, mx = 0, smx = 0;
        for(int i=0;i<=n;i++) {
            res1 = res2 = 0;
            for(int k=1;k<=n;k++)
                res1 = max(res1, a[k][i] + f1[k-1][i] + f2[k+1][i]);
            for(int j=i+1;j<=n+1;j++) {
                res3 = 0;
                for(int k=1;k<=n;k++)
                    res3 = max(res3, a[k][j] + f3[k-1][j] + f4[k+1][j]);
                ret = max(ret, res1 + res2 + res3);
                if( j == n + 1 ) break;
                mx = smx = 0;
                for(int k=1;k<=n;k++) {
                    if( a[k][j] > mx )
                        smx = mx, mx = a[k][j];
                    else if( a[k][j] > smx )
                        smx = a[k][j];
                }
                res2 += (mx + smx);
            }
        }
        return ret;
    }
    int solve2() {
        int ret = 0, res1 = 0, res2 = 0, res3 = 0, mx = 0, smx = 0;
        for(int i=0;i<=n;i++) {
            res1 = res2 = 0;
            for(int k=1;k<=n;k++)
                res1 = max(res1, a[i][k] + f1[i][k-1] + f3[i][k+1]);
            for(int j=i+1;j<=n+1;j++) {
                res3 = 0;
                for(int k=1;k<=n;k++)
                    res3 = max(res3, a[j][k] + f2[j][k-1] + f4[j][k+1]);
                ret = max(ret, res1 + res2 + res3);
                if( j == n + 1 ) break;
                mx = smx = 0;
                for(int k=1;k<=n;k++) {
                    if( a[j][k] > mx )
                        smx = mx, mx = a[j][k];
                    else if( a[j][k] > smx )
                        smx = a[j][k];
                }
                res2 += (mx + smx);
            }
        }
        return ret;
    }
    int solve3() {
        int ret = 0;
        for(int i=1;i<n;i++)
            for(int j=i+1;j<=n;j++)
                for(int k=1;k<n;k++)
                    for(int l=k+1;l<=n;l++) {
                        ret = max(ret, f1[i][l-1] + f2[i+1][k] + f3[j-1][l] + f4[j][k+1]);
                        ret = max(ret, f1[j-1][k] + f2[j][l-1] + f3[i][k+1] + f4[i+1][l]);
                    }
        return ret;
    }
    int maxSum(vector<string>field) {
        n = field.size();
        memset(a, 0, sizeof a);
        for(int i=0;i<n;i++)
            for(int j=0;j<n;j++)
                a[i + 1][j + 1] = field[i][j] - '0';
        get_lu(), get_ld(), get_ru(), get_rd();
        return max(max(solve1(), solve2()), solve3());
    }
}ED;

@details@

写得来,但就是想不到。。。

写的时候 Ctrl + c 与 Ctrl + v 用得很爽 2333。

猜你喜欢

转载自www.cnblogs.com/Tiw-Air-OAO/p/11332953.html