4777: 方格取数 

Time Limit(Common/Java):1000MS/3000MS     Memory Limit:65536KByte
Total Submit: 11            Accepted:10

Description

设有N*N的方格图(N<=10,我们将其中的某些方格中填入正整数,而其他的方格中则放入数字0。如下图所示(见样例):

某人从图的左上角的A 点出发,可以向下行走,也可以向右走,直到到达右下角的B点。在走过的路上,他可以取走方格中的数(取走后的方格中将变为数字0)。

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

Input

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

Output

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

Sample Input

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

Sample Output

67

我觉得大家第一反应应该都是做一次DP之后再做一次,但是这是在贪心,这次走完最大值的路径下次就不走了

所以其实要一起考虑,算法复杂度是n^3,它是有四个点的,包括前一次经过的点,而且步数的几个点是确定的,在一条斜线上

#include <bits/stdc++.h>
using namespace std;
#define max4(a, b, c, d) max(max(a, b), max(c, d))
int dp[][][];
int a[][];
int main()
{
int n, x, y, z;
scanf("%d", &n);
while (~scanf("%d%d%d", &x, &y, &z) && (x || y || z))a[x][y] = z;
for (int k = ; k < n * ; k++)
for (int i = ; i <= n; i++)
for (int j = ; j <= n; j++)
{
if (i > k || j > k || k - j + > n || k - i + > n)continue;
dp[k][i][j] = max4(dp[k - ][i][j], dp[k - ][i - ][j], dp[k - ][i][j - ], dp[k - ][i - ][j - ]);
dp[k][i][j] += a[k - j + ][j];
if (i != j)dp[k][i][j] += a[k - i + ][i];
}
cout << dp[n * - ][n][n];
return ;
}

升级版的k步,需要使用最大流求解

首先拆点,将一个点拆成x和y,然后从x到y连一条容量为1,流量为x的边

然后再连一条容量为inf,费用为0的边

这样即可保证一个点可以走多次,而数只能取一次。然后连接a和b时,从a的y向b的x连一条容量为inf,费用为0的边。最后跑最大费用最大流即可。

#include <bits/stdc++.h>
using namespace std; const int N = 1e4 + ;
const int M = 1e5 + ;
const int INF = 0x3f3f3f3f; int FIR[N], TO[M], CAP[M], FLOW[M], COST[M], NEXT[M], tote;
int pre[N], dist[N], q[];
bool vis[N];
int n, k, S, T;
void init()
{
tote = ;
memset(FIR, -, sizeof(FIR));
}
void add(int u, int v, int cap, int cost)
{
TO[tote] = v;
CAP[tote] = cap;
FLOW[tote] = ;
COST[tote] = cost;
NEXT[tote] = FIR[u];
FIR[u] = tote++; TO[tote] = u;
CAP[tote] = ;
FLOW[tote] = ;
COST[tote] = -cost;
NEXT[tote] = FIR[v];
FIR[v] = tote++;
}
bool SPFA(int s, int t)
{
memset(dist, INF, sizeof(dist));
memset(vis, false, sizeof(vis));
memset(pre, -, sizeof(pre));
dist[s] = ;
vis[s] = true;
q[] = s;
int head = , tail = ;
while (head != tail)
{
int u = q[++head];
vis[u] = false;
for (int v = FIR[u]; v != -; v = NEXT[v])
{
if (dist[TO[v]] > dist[u] + COST[v] && CAP[v] > FLOW[v])
{
dist[TO[v]] = dist[u] + COST[v];
pre[TO[v]] = v;
if (!vis[TO[v]])
{
vis[TO[v]] = true;
q[++tail] = TO[v];
}
}
}
}
return pre[t] != -;
}
void MCMF(int s, int t, int &cost, int &flow)
{
flow = cost = ;
while (SPFA(s, t))
{
int Min = INF;
for (int v = pre[t]; v != -; v = pre[TO[v ^ ]])
Min = min(Min, CAP[v] - FLOW[v]);
for (int v = pre[t]; v != -; v = pre[TO[v ^ ]])
{
FLOW[v] += Min;
FLOW[v ^ ] -= Min;
cost += COST[v] * Min;
}
flow += Min;
}
}
int main()
{
int ca;
cin>>ca;
while(ca--)
{
init();
cin >> n >> k;
S = ;
T = n * n * + ;
int ff=n*n;
for (int i = ; i <= n; i++)
for (int j = , x; j <= n; j++)
{
cin>>x;
int a=(i-)*n+j,b=a+,c=a+n;
add(a,ff+a,INF,),add(a,ff+a,,-x);
if(j<n)add(ff+a,b,INF,);
if(i<n)add(ff+a,c,INF,);
}
add(S, , k, );
add(T - , T, k, );
int cost, flow;
MCMF(S, T, cost, flow);
printf("%d\n", -cost);
}
return ;
}

TZOJ4777: 方格取数的更多相关文章

  1. HDU 1565&1569 方格取数系列(状压DP或者最大流)

    方格取数(2) Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total S ...

  2. NOIP200003方格取数

    NOIP200003方格取数 难度级别: D: 编程语言:不限:运行时间限制:1000ms: 运行空间限制:51200KB: 代码长度限制:2000000B 试题描述 XYZ 是首师大附中信息技术团编 ...

  3. vijos 1563 疯狂的方格取数

    P1653疯狂的方格取数 Accepted 标签:天才的talent[显示标签]   背景 Due to the talent of talent123,当talent123做完NOIP考了两次的二取 ...

  4. [HDU 1565+1569] 方格取数

    HDU 1565 方格取数(1) Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others ...

  5. 网络流(最大流) HDU 1565 方格取数(1) HDU 1569 方格取数(2)

      HDU 1565 方格取数(1) 给你一个n*n的格子的棋盘,每个格子里面有一个非负数.从中取出若干个数,使得任意的两个数所在的格子没有公共边,就是说所取的数所在的2个格子不能相邻,并且取出的数的 ...

  6. HDU-1565 方格取数(1)

    http://acm.hdu.edu.cn/showproblem.php?pid=1565 方格取数(1) Time Limit: 10000/5000 MS (Java/Others)    Me ...

  7. BZOJ 1475: 方格取数( 网络流 )

    本来想写道水题....结果调了这么久!就是一个 define 里面少加了个括号 ! 二分图最大点权独立集...黑白染色一下 , 然后建图 : S -> black_node , white_no ...

  8. [动态规划]P1004 方格取数

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

  9. P2045 方格取数加强版

    P2045 方格取数加强版 题目描述 给出一个n*n的矩阵,每一格有一个非负整数Aij,(Aij <= 1000)现在从(1,1)出发,可以往右或者往下走,最后到达(n,n),每达到一格,把该格 ...

随机推荐

  1. PHP实现记录浏览历史页面

    <?php /******* 说明:cookie只能保存字符串 本实例中,需要保存多个URL(历史访问记录),思路是先将URL数组转为字符串,然后保存,读取时,再循环读取 *******/ // ...

  2. XML 映射文件

    MyBatis 的真正强大在于它的映射语句,这是它的魔力所在.由于它的异常强大,映射器的 XML 文件就显得相对简单.如果拿它跟具有相同功能的 JDBC 代码进行对比,你会立即发现省掉了将近 95% ...

  3. Vue 获取dom元素之 ref 和 $refs 详解

    一.$refs 一个对象,持有ref注册过的所有元素或子组件.(注册过的 ref 的集合) 二.ref 被用来给元素或子组件注册引用信息.若用在dom元素上,引用指向的就是dom元素:若用在子组件上, ...

  4. 转载 Python 安装setuptools和pip工具操作方法(必看)

    本文章转载自 脚本之家 http://www.jb51.net  感谢! setuptools模块和pip模块是python进行第三方库扩展的极重要工具,例如我们在需要安装一些爬虫或者数据分析的包时就 ...

  5. BTC功能类

    <?php/*EasyBitcoin-PHP A simple class for making calls to Bitcoin's API using PHP.https://github. ...

  6. 在vue项目中正确的引入jquery

    最近学习vue,习惯性的通过<script>标签引入jquery,写完后报错才想起来,这种方式在vue是不适用的. 1:因为已经安装了vue脚手架,所以需要在webpack中全局引入jqu ...

  7. H5C3--设置颜色的几种方式

    设置颜色的方式: 关键字:red|blue 第一种:十六进制:#ffffff 第二种:rgb(红,绿,蓝): rgb(ffff00) rgba(红,绿,蓝,透明度) 第三种:hsl(色相,饱和度,明度 ...

  8. CodeChef--SEPT14小结

    这套题目只做了几个相对简单的.其他的做起来比较吃力. A 找下规律 /***************************************************************** ...

  9. DLX

    #include <iostream> #include <cstdlib> #include <cstring> #include <queue> # ...

  10. UML类图解释

    那个动物矩形框,它就代表一个类(Class).类图分三层,第一层显示类的名称,如果是抽象类,则就用斜体显示.第二层是类的特性,通常是字段和属性.第三层是类的操作,通常是方法或行为.注意前面的符号,“+ ...