算法训练 方格取数

ALGO-19 方格取数

资源限制

时间限制:1.0s 内存限制:256.0MB

问题描述

设有N*N的方格图(N<=10),我们将其中的某些方格中填入正整数,而其他的方格中则放入数字0。
某人从图的左上角的A 点(1,1)出发,可以向下行走,也可以向右走,直到到达右下角的B点(N,N)。在走过的路上,他可以取走方格中的数(取走后的方格中将变为数字0)。
此人从A点到B 点共走两次,试找出2条这样的路径,使得取得的数之和为最大。

输入格式

输入的第一行为一个整数N(表示N*N的方格图),接下来的每行有三个整数,前两个表示位置,第三个数为该位置上所放的数。一行单独的0表示输入结束。

输出格式

只需输出一个整数,表示2条路径上取得的最大的和。

测试样例

输入:
8
2 3 13
2 6 6
3 5 7
4 4 14
5 2 21
5 6 4
6 3 15
7 2 14
0 0 0

输出:
67  

数据规模与约定

N<=10

AC 在下面
战略性尝试 (WA)

import java.io.*;
import java.util.*;

public class Main {

    static int[][] map;

    public static void main(String[] args) {
        InputReader in = new InputReader(System.in);
        int N = in.nextInt(), x;
        map = new int[N + 1][N + 1];
        do map[x = in.nextInt()][in.nextInt()] = in.nextInt();
        while (x != 0);
        System.out.print(full(N, true) + full(N, false));
    }

    static int full(int N, boolean flag) {
        int dp[][] = new int[N + 1][N + 1];
        for (int i = 1; i <= N; i++)
            for (int j = 1; j <= N; j++) dp[i][j] = Math.max(dp[i][j - 1], dp[i - 1][j]) + map[i][j];
        if (flag) {
            List<Integer> list = new ArrayList();
            list.add(dp[N][N]);
            for (int k = N; k >= 1; k--) {
                for (int i = N; i >= k; i--)
                    for (int j = N; j >= k; j--) {
                        if (map[i][j] == 0) continue;
                        if (list.contains(dp[i][j])) {
                            list.add(dp[i][j] - map[i][j]);
                            list.remove(new Integer(dp[i][j]));
                            map[i][j] = 0;
                        }
                    }
            }
        }
        return dp[N][N];
    }

    static class InputReader { . . . }
}

在这里插入图片描述
仔细看了下 测试用例三

7 1 3 2 1 4 3 2 3 3 3 3 3 5 5 4 6 5 4 7 3 2 7 5 4 0 0 0

在这里插入图片描述
权值和最大的两条路线,不能被分解成一条权值和最大一条次之的路线

于是它就成了套娃题

AC code:
import java.io.*;
import java.util.*;

public class Main {

    public static void main(String[] args) {
        InputReader in = new InputReader(System.in);
        int N = in.nextInt() + 1, X;
        int[][] map = new int[N][N], dp[][] = new int[N][N][N][N];
        do map[X = in.nextInt()][in.nextInt()] = in.nextInt();
        while (X != 0);
        for (int i = 1; i < N; i++)
            for (int j = 1; j < N; j++)
                for (int x = 1; x < N; x++)
                    for (int y = 1; y < N; y++)
                        dp[i][j][x][y] =
                            max(max(dp[i - 1][j][x - 1][y], dp[i][j - 1][x - 1][y]), max(dp[i][j - 1][x][y - 1], dp[i - 1][j][x][y - 1]))
                            + (i == x && j == y? map[i][j]: map[i][j] + map[x][y]);
        System.out.print(dp[--N][N][N][N]);
    }

    static int max(int a, int b) { return a > b? a: b; }

    static class InputReader { . . . }
}
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值