HLG 2163 方格取数 (最大网络流)
题目链接: m=ProblemSet&a=showProblem&problem_id=2163">点击打开链接
Description :
给你一个n*n的格子的棋盘,每一个格子里面有一个非负数。如今从中取出若干个数,使得随意的两个数所在的格子没有公共边,就是说所取的数所在的2个格子不能相邻。而且取出的数的和最大。
Input :
包含多个測试实例,每一个測试实例包含一个整数n 和n*n个非负数x(n<=20, 0 <= x <= 1000)。
Output :
对于每一个測试实例。输出可能取得的最大的和。
Sample Input :
3
258 83 905
874 941 662
733 415 890
Sample Output :
3727
解析:
一開始的方法(代码):
#include <iostream>
#include <cstdio>
#include <cstring>
#include <cstdlib>
#define MAXN 25
#define RST(N)memset(N, 0, sizeof(N))
using namespace std; int n, res, Max;
int Map[MAXN][MAXN], vis[MAXN][MAXN];
const int dx[] = {-1, 1, 1, -1};
const int dy[] = {1, 1, -1, -1}; int max(int x, int y) { return x>y ? x:y; } bool check(int x, int y)
{
return x>=1&&x<=n&&y>=1&&y<=n&&!vis[x][y];
} void solve(int px, int py)
{
int xx, yy;
vis[px][py] = 1;
//printf("%d(px), %d(py)\n", px, py);
for(int i=0; i<4; i++) {
xx = px+dx[i];
yy = py+dy[i];
if(check(xx, yy)) {
//printf("%d(xx), %d(yy) is usable\n", xx, yy);
solve(xx, yy);
}
//else printf("%d(xx), %d(yy) is XXXXXXX\n", xx, yy);
}
res += Map[px][py];
//printf("res = %d\n", res);
} int main()
{
while(~scanf("%d", &n)) {
Max = -1;
for(int i=1; i<=n; i++) {
for(int j=1; j<=n; j++) {
scanf("%d", &Map[i][j]);
}
}
if(n == 1) printf("%d\n", Map[1][1]);
else if(n == 2) {
printf("%d\n", max(Map[1][1]+Map[2][2], Map[1][2]+Map[2][1]));
}else {
for(int i=1; i<=n; i++) {
for(int j=1; j<=n; j++) {
res = 0;
RST(vis);
solve(i, j);
if(res > Max) Max = res;
}
}
printf("%d\n", Max);
}
}
return 0;
}
一開始想到的方法感觉非常对非常对。然后过了几天又一次做的时候突然想到一个过不去的情况,所以就想到了网络流;
一開始用的是深搜。思路是:
watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQva2VzaGFjb29raWU=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" alt="">
遍历二维数组中的每个点,然后递归遍历每个和当前点无重边的点。这样的方法能測试到大多数情况;但还是菜鸟思想。由于好久没怎么刷题了,思维有点跟不上了;这样的方法对于下面这样的情况就測试不正确了:
假设输入数据:
3
520 10 45
10 70 600
10 60 55
按上述方法的话会出现下面两种情况:
最后出现两种结果: 520+70+45+55+10 = 700 10+600+60+10=680
这两种答案都是错误的,由于正确的应该是: 520 + 600 + 60 = 1180
watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQva2VzaGFjb29raWU=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" alt="">
这个题因为数据范围较大,所以状态压缩过不去,须要用网络流,我反复一遍建图:
我们知道对于普通二分图来说,最大独立点集 + 最小点覆盖集 = 总点数,类似的,对于有权的二分图来说。有:
最大点权独立集 + 最小点权覆盖集 = 总点权和。
这个题非常明显是要求 最大点权独立集 。如今 总点权 已知,我们仅仅要求出来 最小点权覆盖集 就好了。我们能够这样建图,
1,对矩阵中的点进行黑白着色(相邻的点颜色不同),从源点向黑色的点连一条边,权值为该黑色点的权值。
2,从白色的点向汇点连一条边,权值为该白色点的权值,
3。然后。对于每一对相邻的黑白点,从黑点向白点连一条边。权值为无穷大。
最后求最小割(最大流)。即为最小点权覆盖集。
由于我们求出的最小割集一定是从那些相邻的黑白点之间的边(也就是不能用的边。由于相邻的数不能同一时候选)中选出来的,且是最小代价,也就是说从方格中拿掉的数之和尽量小,那么剩下的数之和一定是最大的。
代码例如以下:
#include <iostream>
#include <cstdio>
#include <cstring>
#include <queue>
#include <cstdlib>
#include <algorithm>
#define VM 2520
#define EM 500050
#define INF 0x3f3f3f3f
#define RST(N)memset(N, 0, sizeof(N))
using namespace std; struct Edge
{
int u, v, nxt;
int flow;
}edge[EM << 1]; int n, m, cnt, head[VM];
int src, des, dep[VM]; void addedge(int cu, int cv, int cf)
{
edge[cnt].u = cu, edge[cnt].v = cv, edge[cnt].flow = cf;
edge[cnt].nxt = head[cu], head[cu] = cnt++; edge[cnt].u = cv, edge[cnt].v = cu, edge[cnt].flow = 0;
edge[cnt].nxt = head[cv], head[cv] = cnt++;
} int dir[4][2]= {{-1, 0}, {1, 0}, {0, -1}, {0, 1}}; int legal(int i, int j, int k)
{
int x = i + dir[k][0];
int y = j + dir[k][1];
return x>=1 && x<=n && y>=1 && y<=m;
} int BFS() //又一次建图(按层数建图)
{
queue <int> q;
while(!q.empty()) q.pop();
memset(dep, -1, sizeof(dep));
dep[src] = 0;
q.push(src);
while(!q.empty()) {
int u = q.front();
q.pop();
for(int i=head[u]; i!=-1; i=edge[i].nxt) {
int v = edge[i].v;
if(edge[i].flow>0 && dep[v]==-1) { // 假设能够到达但还没有訪问
dep[v] = dep[u] + 1;
q.push(v);
}
}
}
return dep[des] != -1;
} /*
int DFS(int u,int minx) //查找路径上的最小的流量
{
if(u == des) return minx;
int tmp;
for(int i=head[u]; i!=-1; i=edge[i].nxt) {
int v = edge[i].v;
if(edge[i].flow>0 && dep[v]==dep[u]+1 && (tmp=DFS(v, min(minx, edge[i].flow)))) {
edge[i].flow -= tmp;
edge[i^1].flow += tmp;
return tmp;
}
}
return 0;
}
*/ int DFS(int u,int minx)
{
int ans = 0;
if(u == des) return minx;
for(int i=head[u]; i!=-1 && ans<minx; i=edge[i].nxt) {
int v = edge[i].v;
if(edge[i].flow>0 && dep[v]==dep[u]+1){
int tmp = min(edge[i].flow, minx-ans);
tmp = DFS(v,tmp);
ans += tmp;
edge[i].flow -= tmp;
edge[i^1].flow += tmp;
}
}
if(!ans) dep[u] = -2;
return ans;
} int Dinic()
{
int ans = 0, tmp;
while(BFS()) {
while(1) {
tmp = DFS(src, INF);
if(tmp == 0) break;
ans += tmp;
}
}
return ans;
} void Init()
{
m = n;
cnt = src = 0;
des = n * m + 1;
memset(head, -1, sizeof(head));
} int main(int argc, char **argv)
{
while(~scanf("%d", &n)) {
Init();
int x, sum = 0;
for(int i=1; i<=n; i++) {
for(int j=1; j<=m; j++) {
scanf("%d", &x);
sum += x;
if((i+j)%2 == 0) {
addedge(src, (i-1)*m+j, x);
for(int k=0; k<4; k++) {
if(legal(i, j, k)) addedge((i-1)*m+j, (i+dir[k][0]-1)*m+(j+dir[k][1]), INF);
}
}else {
addedge((i-1)*m+j, des, x);
for(int k=0; k<4; k++) {
if(legal(i, j, k)) addedge((i+dir[k][0]-1)*m+(j+dir[k][1]), (i-1)*m+j, INF);
}
}
}
}
int maxflow = Dinic();
printf("%d\n", sum-maxflow);
}
return 0;
}
HLG 2163 方格取数 (最大网络流)的更多相关文章
- P2774 方格取数问题 网络流重温
P2774 方格取数问题 这个题目之前写过一次,现在重温还是感觉有点难,可能之前没有理解透彻. 这个题目要求取一定数量的数,并且这些数在方格里面不能相邻,问取完数之后和最大是多少. 这个很好的用了网络 ...
- P2774 方格取数问题 网络流
题目: P2774 方格取数问题 题目背景 none! 题目描述 在一个有 m*n 个方格的棋盘中,每个方格中有一个正整数.现要从方格中取数,使任意 2 个数所在方格没有公共边,且取出的数的总和最大. ...
- CODEVS_1227 方格取数2 网络流 最小费用流 拆点
原题链接:http://codevs.cn/problem/1227/ 题目描述 Description 给出一个n*n的矩阵,每一格有一个非负整数Aij,(Aij <= 1000)现在从(1, ...
- LibreOJ #6007. 「网络流 24 题」方格取数 最小割 最大点权独立集 最大流
#6007. 「网络流 24 题」方格取数 内存限制:256 MiB时间限制:1000 ms标准输入输出 题目类型:传统评测方式:文本比较 上传者: 匿名 提交提交记录统计讨论测试数据 题目描述 ...
- Libre 6007 「网络流 24 题」方格取数 / Luogu 2774 方格取数问题 (网络流,最大流)
Libre 6007 「网络流 24 题」方格取数 / Luogu 2774 方格取数问题 (网络流,最大流) Description 在一个有 m*n 个方格的棋盘中,每个方格中有一个正整数.现要从 ...
- HDU 1565 - 方格取数(1) - [状压DP][网络流 - 最大点权独立集和最小点权覆盖集]
题目链接:https://cn.vjudge.net/problem/HDU-1565 Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 32 ...
- AC日记——[网络流24题]方格取数问题 cogs 734
734. [网络流24题] 方格取数问题 ★★☆ 输入文件:grid.in 输出文件:grid.out 简单对比时间限制:1 s 内存限制:128 MB «问题描述: 在一个有m*n ...
- Cogs 734. [网络流24题] 方格取数问题(最大闭合子图)
[网络流24题] 方格取数问题 ★★☆ 输入文件:grid.in 输出文件:grid.out 简单对比 时间限制:1 s 内存限制:128 MB «问题描述: 在一个有m*n 个方格的棋盘中,每个方格 ...
- 网络流(最大流) HDU 1565 方格取数(1) HDU 1569 方格取数(2)
HDU 1565 方格取数(1) 给你一个n*n的格子的棋盘,每个格子里面有一个非负数.从中取出若干个数,使得任意的两个数所在的格子没有公共边,就是说所取的数所在的2个格子不能相邻,并且取出的数的 ...
随机推荐
- TPersistent的三个用途(读写DFM文件,Assign,RTTI),最主要还是第三个用途
不是什么类对象都需要RTTI,如果把它放在TObject,除了增加可执行文件的大小以及运行内存空间以外,没什么好处.
- linux 下vi中关于删除某段,某行,或者全部删除的命令
1,先打开某个文件: vi filename 2,转到文件结尾 在命令模式输入 G 转到10行 在命令模式输入 10G 4,删除所有内容:先用G 转到文件尾,然后使用下面命令: :1, ...
- Netty源代码学习——ChannelPipeline模型分析
參考Netty API io.netty.channel.ChannelPipeline A list of ChannelHandlers which handles or intercepts i ...
- Android中贝塞尔曲线的绘制方法
贝塞尔曲线,很多人可能不太了解,什么叫做贝塞尔曲线呢?这里先做一下简单介绍:贝塞尔曲线也可以叫做贝济埃曲线或者贝兹曲线,它由线段与节点组成,节点是可拖动的支点,线段像可伸缩的皮筋.一般的矢量图形软件常 ...
- SDUT 2893-B(DP || 记忆化搜索)
B Time Limit: 1000ms Memory limit: 65536K 有疑问?点这里^_^ 题目描写叙述 有n块地板排成一条直线,从左到右编号为1,2,3. . . n-1,n,每 ...
- tomcat加载时报The web application [/dmscs] created a ThreadLocal with key of type
严重: The web application [/dmscs] created a ThreadLocal with key of type [com.opensymphony.xwork2.inj ...
- 用SignalR做类似QQ登录的应用
原文:用SignalR做类似QQ登录的应用 首先通过NuGet下载signalr包 在工程下新建一个类,继承Hub public class DemoHub:Hub { public class Us ...
- Linux注销在线用户
与Windows系统类似,Linux系统上也有注销在线用户的方法,我们可以使用pkill命令,详细的步骤如下: . 首先使用w或who命令查看在线用户,确定用户所在TTY [root@iavp232 ...
- Makefile自动生成工具-----autotools的使用(详细)
相信每个学习Linux的人都知道Makefile,这是一个很有用的东西,但是编写它是比较复杂,今天介绍一个它的自动生成工具,autotools的使用.很多GNULinux的的软件都是用它生成Makef ...
- Swift - 发送消息(文本,图片,文件等)给微信好友或分享到朋友圈
通过调用微信提供的API接口,我们可以很方便的在应用中发送消息给微信好友,或者分享到朋友圈.在微信开发平台(https://open.weixin.qq.com)里,提供了详细的说明文档和样例.但由于 ...