题目链接:https://www.luogu.org/problemnew/show/P1004

题目描述

设有N*N的方格图(N<=9),我们将其中的某些方格中填入正整数,而其他的方格中则放

人数字0。如下图所示(见样例):

A
0 0 0 0 0 0 0 0
0 0 13 0 0 6 0 0
0 0 0 0 7 0 0 0
0 0 0 14 0 0 0 0
0 21 0 0 0 4 0 0
0 0 15 0 0 0 0 0
0 14 0 0 0 0 0 0
0 0 0 0 0 0 0 0
. B

某人从图的左上角的A点出发,可以向下行走,也可以向右走,直到到达右下角的B

点。在走过的路上,他可以取走方格中的数(取走后的方格中将变为数字0)。

此人从A点到B点共走两次,试找出2条这样的路径,使得取得的数之和为最大。

输入输出格式

输入格式:

输入的第一行为一个整数N(表示N*N的方格图),接下来的每行有三个整数,前两个

表示位置,第三个数为该位置上所放的数。一行单独的0表示输入结束。

输出格式:

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

输入输出样例

输入样例#1: 

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
输出样例#1: 

67
四维dp数组:
将这两条路线看成是两个人同时走不同的路线,开一个四维dp数组记录下这两人的坐标。
#include <bits/stdc++.h>
using namespace std;
int mapp[][];
int dp[][][][]; //两条路线,既可以假设为两个人同时走不同的路线,dp[i][j][k][l]表示这两个人走到该点取数的最大和
//mapp[i][j]为第一人的坐标,mapp[k][l]为第二人的坐标
int main()
{
int n, x, y, m;
cin >>n>> x >> y >> m;
while (x != || y != || m != )
{
mapp[x][y] = m;
cin >> x >> y >> m;
}
for (int i = ; i <= n; i++)
for (int j = ; j <= n; j++)
for(int k=;k<=n;k++)
for (int l = ; l <= n; l++)
{ //由于任意一点mapp[i][j]只能由mapp[i-1][j],或者mapp[i][j-1]达到,且控制这两个人的步数相同(即i和j中有且仅有一个数-1,k和l中有且仅有一个数-1),所以总共由4种情况要考虑
dp[i][j][k][l] = max(max(dp[i][j-][k-][l], dp[i][j - ][k][l-]), max(dp[i-][j][k - ][l], dp[i-][j][k][l - ])) + mapp[i][j] + mapp[k][l];
if (i == k && j == l)dp[i][j][k][l] -= mapp[i][j]; //由于方格中的数只能取一次,所以如果这两个人走到了同一个点,mapp[i][j]只能加一次
}
cout << dp[n][n][n][n];
return ;
}
接下来是对以上四维dp的优化,三维dp数组也能解决

定义d[i][j][k]为走了i步,路径a向下走了j步,路径b向下了k步,取数和的最大值;

此时(i-j),(i-k)就是a与b此时到达点的纵坐标;其它的地方都与四维dp类似

#include <bits/stdc++.h>
using namespace std;
int mapp[][];
int f[][][];
int n, x, y, z;
int main()
{
scanf("%d", &n);
memset(f, , sizeof(f));
memset(mapp, , sizeof(mapp));
while (scanf("%d%d%d", &x, &y, &z), x || y || z)
mapp[x][y] = z;
for (int k = ; k <= * n - ; k++)
{
for (int i = ; i <= min(k, n); i++)
{
for (int j = ; j <= min(n, k); j++)
{
f[k][i][j] = max(max(f[k - ][i][j], f[k - ][i][j - ]), max(f[k - ][i - ][j], f[k - ][i - ][j - ]));
if (i == j) f[k][i][j] += mapp[i][k - j + ];
else f[k][i][j] += mapp[i][k - i + ] + mapp[j][k - j + ];
}
}
}
printf("%d\n", f[ * n - ][n][n]);
return ;
}
2018-05-15

洛谷 P1004 方格取数 【多进程dp】的更多相关文章

  1. 洛谷P1004 方格取数-四维DP

    题目描述 设有 N \times NN×N 的方格图 (N \le 9)(N≤9) ,我们将其中的某些方格中填入正整数,而其他的方格中则放入数字 00 .如下图所示(见样例): A 0 0 0 0 0 ...

  2. 洛谷 - P1004 - 方格取数 - 简单dp

    https://www.luogu.org/problemnew/show/P1004 这道题分类到简单dp但是感觉一点都不简单……这种做两次的dp真的不是很懂怎么写.假如是贪心做两次,感觉又不能证明 ...

  3. 棋盘DP三连——洛谷 P1004 方格取数 &&洛谷 P1006 传纸条 &&Codevs 2853 方格游戏

    P1004 方格取数 题目描述 设有N $\times N$N×N的方格图(N $\le 9$)(N≤9),我们将其中的某些方格中填入正整数,而其他的方格中则放入数字00.如下图所示(见样例): A ...

  4. 洛谷 P1004 方格取数 题解

    P1004 方格取数 题目描述 设有 \(N \times N\) 的方格图 \((N \le 9)\),我们将其中的某些方格中填入正整数,而其他的方格中则放入数字\(0\).如下图所示(见样例): ...

  5. 洛谷P1004 方格取数

    网络流大法吼 不想用DP的我选择了用网络流-- 建模方法: 从源点向(1,1)连一条容量为2(走两次),费用为0的边 从(n,n)向汇点连一条容量为2,费用为0的边 每个方格向右边和下边的方格连一条容 ...

  6. 洛谷 P1004 方格取数 【多线程DP/四维DP/】

    题目描述(https://www.luogu.org/problemnew/show/1004) 设有N*N的方格图(N<=9),我们将其中的某些方格中填入正整数,而其他的方格中则放 人数字0. ...

  7. 洛谷 P1004 方格取数

    题目描述 设有N*N的方格图(N<=9),我们将其中的某些方格中填入正整数,而其他的方格中则放 人数字0.如下图所示(见样例): A 0 0 0 0 0 0 0 0 0 0 13 0 0 6 0 ...

  8. 【动态规划】洛谷P1004方格取数

    题目描述 设有N*N的方格图(N<=9),我们将其中的某些方格中填入正整数,而其他的方格中则放 人数字0.如下图所示(见样例): A 0 0 0 0 0 0 0 0 0 0 13 0 0 6 0 ...

  9. 四维动规 洛谷P1004方格取数

    分析:这个题因为数据量非常小,可以直接用四维的DP数组 dp[i][j][k][l]表示第一个人走到位置(i,j),第二个人走到位置[k][l]时所取的数的最大和 状态转移方程可以轻松得出为:dp[i ...

随机推荐

  1. Activity生命周期函数、onSaveInstanceState()和onRestoreInstanceState()的介绍

    http://www.cnblogs.com/tianzhijiexian/p/3885472.html

  2. Dom4j向XML中增加节点与属性——(四)

    先获取到节点,然后在节点山添加Element 添加节点 添加属性 设置开始标签与结束标签的值book.addElement("描述").addAttribute("nam ...

  3. 【vim】缩写 :ab [缩写] [要替换的文字]

    一个很可能是最令人印象深刻的窍门是你可以在 Vim 中定义缩写,它可以实时地把你输入的东西替换为另外的东西.语法格式如下: :ab [缩写] [要替换的文字] 一个通用的例子是: :ab asap a ...

  4. Linux内核驱动--硬件访问I/O【原创】

    寄存器与内存 寄存器与内存的区别在哪里呢? 寄存器和RAM的主要不同在于寄存器操作有副作用(side effect或边际效果): 读取某个地址时可能导致该地址内容发生变化,比如很多设备的中断状态寄存器 ...

  5. MySQL5.7更改用户名密码

    更改用户名密码,官方推荐使用alter ALTER USER test@'%' IDENTIFIED BY '; 还有一种 update mysql.user set authentication_s ...

  6. java多线程系列六、线程池

    一. 线程池简介 1. 线程池的概念: 线程池就是首先创建一些线程,它们的集合称为线程池. 2. 使用线程池的好处 a) 降低资源的消耗.使用线程池不用频繁的创建线程和销毁线程 b) 提高响应速度,任 ...

  7. DES加密模式详解

    DES加密模式详解 http://www.cnblogs.com/Lawson/archive/2012/05/20/2510781.html http://www.blogjava.net/wayn ...

  8. Win10 + Visual Studio 2017 下 OpenCV无法显示图像的问题

    测试代码如下: #include "stdafx.h" #include<opencv2\opencv.hpp> #include<opencv2\highgui ...

  9. Project Euler Problem6

    Sum square difference Problem 6 The sum of the squares of the first ten natural numbers is, 12 + 22  ...

  10. openwrt git 代码下载地址

    openwrt 各个版本代码下载 trunk:git clone git://github.com/openwrt/openwrt.git 15.05  (Chaos Calmer)git clone ...