Problem Description
给你一个n*n的格子的棋盘,每个格子里面有一个非负数。
从中取出若干个数,使得任意的两个数所在的格子没有公共边,就是说所取的数所在的2个格子不能相邻,并且取出的数的和最大。
 
Input
包括多个测试实例,每个测试实例包括一个整数n 和n*n个非负数(n<=20)
 
Output
对于每个测试实例,输出可能取得的最大的和
 
Sample Input
3
75 15 21
75 15 28
34 70 5
Sample Output
188
Author
ailyanlu
Source
 
 
正解:网络流
解题报告:
  网络流24题第9题。同样是可以转换为最大点权和独立集,求最小割跑最大流。自己的题解:

  模型:建图+最小割

  总结:

  和前面做的几道题很类似,都是求最大独立集。事实上都可以转换成二分图,然后按照最小割模型来做。

  考虑这道题说不能选取有相邻边的格子中的数,相当于是如果选取了当前格子,那么周围相邻的四个格子都不能再选。这个模型显然就是一个二分图,所以首先我们可以先把全图进行二分图染色,对于一个点坐标为(i,j),如果(i+j)%2==0,那么染成白色;否则,染成黑色。然后新建源点S,向所有白色点连一条容量为该点点权的有向边;新建汇点T,所有黑色点连一条容量为该点点权的有向边。对于所有白色点,向四周的所有黑色点都连一条容量为无穷的有向边。如此以来我们构出了一张图(跟前几道题一样的说)。

  显然我们希望得到选取的尽可能多的格子,由于都是正数所以肯定能选尽量选。但是如果选了一个格子,就意味着四周的全都不能再选了,也就是说如果我们把ans初值记为所有格子权值之和,那么我们每次因为选择了一个格子就会失去周围四个的权值,需要从ans中减掉。这跟什么很像呢?最大半连通子图!是的,一样的做法,用总的和减去最小割就可以得到我们期望的最大值。

  
 
 //It is made by jump~
#include <iostream>
#include <cstdlib>
#include <cstring>
#include <cstdio>
#include <cmath>
#include <algorithm>
#include <ctime>
#include <vector>
#include <queue>
#include <map>
#include <set>
#ifdef WIN32
#define OT "%I64d"
#else
#define OT "%lld"
#endif
using namespace std;
typedef long long LL;
const int MAXN = ;
const int MAXM = ;
const int inf = (<<);
int n,sum,S,T,ecnt,ans;
int a[MAXN][MAXN],deep[MAXN*MAXN];
int first[MAXN*MAXN];
queue<int>Q;
struct edge{
int next,to,f;
}e[MAXM]; inline int getint()
{
int w=,q=;
char c=getchar();
while((c<'' || c>'') && c!='-') c=getchar();
if (c=='-') q=, c=getchar();
while (c>='' && c<='') w=w*+c-'', c=getchar();
return q ? -w : w;
} inline void link(int x,int y,int z){
e[++ecnt].next=first[x]; first[x]=ecnt; e[ecnt].to=y; e[ecnt].f=z;
e[++ecnt].next=first[y]; first[y]=ecnt; e[ecnt].to=x; e[ecnt].f=;
} inline bool bfs(){
while(!Q.empty()) Q.pop();
for(int i=;i<=T;i++) deep[i]=;
Q.push(S); deep[S]=;
while(!Q.empty()) {
int u=Q.front(); Q.pop();
for(int i=first[u];i;i=e[i].next) {
int v=e[i].to;
if(e[i].f && !deep[v]) deep[v]=deep[u]+,Q.push(v);
}
}
if(deep[T]!=) return true;
return false;
} inline int Dinic(int x,int remain){
if(remain== || x==T) return remain;
int f,flow=;
for(int i=first[x];i;i=e[i].next) {
int v=e[i].to;
if(e[i].f && deep[v]==deep[x]+){
f=Dinic(v,min(remain,e[i].f));
if(f){
flow+=f; e[i].f-=f; e[i^].f+=f;
remain-=f; if(remain==) return flow;
} else deep[v]=-;
}
}
return flow;
} inline void work(){
while(scanf("%d",&n)!=EOF) {
sum=;
for(int i=;i<=n;i++) for(int j=;j<=n;j++) a[i][j]=getint(),sum+=a[i][j];
S=n*n+;T=S+; ecnt=; memset(first,,sizeof(first));
for(int i=;i<=n;i++) for(int j=;j<=n;j++) if((i+j)%==) link(S,(i-)*n+j,a[i][j]); else link((i-)*n+j,T,a[i][j]);//黑白染色
for(int i=;i<=n;i++)
for(int j=;j<=n;j++){
if((i+j)%) continue;
if(i!=) link((i-)*n+j,(i-)*n+j,inf);
if(j!=) link((i-)*n+j,(i-)*n+j-,inf);
if(j!=n) link((i-)*n+j,(i-)*n+j+,inf);
if(i!=n) link((i-)*n+j,i*n+j,inf);
}
ans=;
while(bfs()) ans+=Dinic(S,inf);
printf("%d\n",sum-ans);
}
} int main()
{
work();
return ;
}

HDU1565 方格取数(1)的更多相关文章

  1. HDU1565 方格取数(1) —— 状压DP or 插头DP(轮廓线更新) or 二分图点带权最大独立集(最小割最大流)

    题目链接:https://vjudge.net/problem/HDU-1565 方格取数(1) Time Limit: 10000/5000 MS (Java/Others)    Memory L ...

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

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

  3. HDU1565 方格取数 &&uva 11270 轮廓线DP

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

  4. Hdu-1565 方格取数(1) (状态压缩dp入门题

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

  5. HDU1565 方格取数1(构图+网络流最大独立集合)

    题目大意:给你一个n*n的格子的棋盘,每个格子里面有一个非负数. 从中取出若干个数,使得任意的两个数所在的格子没有公共边,就是说所取的数所在的2个格子不能相邻,并且取出的数的和最大. 解题思路:最大点 ...

  6. HDU1565方格取数

    典型的状态压缩DP问题.第i行的取法只受到第i-1行的影响.首先每一行的取法要相容(不能有两个相邻),然后相邻行之间也要相容.将每一个格子看做两种状态,1表示取,0表示不取.这样每一行就是一个01串, ...

  7. HDU1565 方格取数(1)(状态压缩dp)

    题目链接. 分析: 说这题是状态压缩dp,其实不是,怎么说呢,题目数据太水了,所以就过了.手动输入n=20的情况,超时.正解是网络流,不太会. A这题时有个细节错了,是dp[i][j]还是dp[i][ ...

  8. HDU 1565 - 方格取数(1) - [状压DP][网络流 - 最大点权独立集和最小点权覆盖集]

    题目链接:https://cn.vjudge.net/problem/HDU-1565 Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 32 ...

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

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

随机推荐

  1. influxDB---Data Exploration

    the group clause group by 返回的分组结果是根据用户指定的tag ,time interval. 1.group by tags 2.group by time interva ...

  2. JavaSctipt语句for循环的思考

    一.语法 for (语句 1; 语句 2; 语句 3) { 被执行的代码块; } 语句 1: 在循环(代码块)开始前执行, 可选. 语句 2: 定义运行循环(代码块)的条件, 可选, 如果省略了语句 ...

  3. jQuery方法find()与children()区别

    一.find() 1.1 说明 find()方法返回被选元素的后代元素,一路向下直到最后一个后代. 1.2 示例 <div> <p> <span>1</spa ...

  4. setlocale同mbstowcs函数的关系(VS2008下setlocale(LC_ALL, "chs")可以执行成功,BCB使用setlocale(LC_ALL, "Chinese (Simplified)_People's Republic of China"),linux上locale别名表大概在 /usr/lib/X11/locale/locale.alias)

    序中,如果要将ASCII码字符串转换为宽字符(Unicode),可以利用标准C的mbstowcs函数. 微软在MSDN中有示例,如下: 然而,这段代码在处理含有汉字的字符串时就会出现问题.比如将: w ...

  5. MySQL中有关icp mrr和bka的特性

    文辉考我的问题,有关这三个的特性,如果在面试过程中,个人见解可以答以下 icp MyQL数据库会在取出索引的同时,判断是否进行WHERE条件过滤,也就是把WHERE的部分过滤操作放在存储引擎层,在某些 ...

  6. django自带的用户认证和form表单功能

    一.用户认证 1.用户认证方法 1.ajango自带用户认证功能,只需要引入相应的模块就可以使用,但是前提是必须使用ajango自带的auth_user表,并且需要把用户相关信息存放在该表中. 2.引 ...

  7. C#类和结构(1)

    1.结构功能特性? 实现代码? 结构用struct关键字定义的,与类类似,但有本质区别.结构实质是一个值类型,它不需要对分配的. 结构的特性: (1).结构作为参数传递时,是值传递. (2).结构的构 ...

  8. LeetCode:旋转图像【48】

    LeetCode:旋转图像[48] 题目描述 给定一个 n × n 的二维矩阵表示一个图像. 将图像顺时针旋转 90 度. 说明: 你必须在原地旋转图像,这意味着你需要直接修改输入的二维矩阵.请不要使 ...

  9. LeetCode:寻找数组的中心索引【668】

    LeetCode:寻找数组的中心索引[668] 题目描述 给定一个整数类型的数组 nums,请编写一个能够返回数组“中心索引”的方法. 我们是这样定义数组中心索引的:数组中心索引的左侧所有元素相加的和 ...

  10. 【数学建模】MATLAB学习笔记——函数式文件

    MATLAB学习笔记——函数式文件 引入函数式文件 说明: 函数式文件主要用于解决计算中的参数传递和函数调用的问题. 函数式的标志是它的第一行为function语句. 函数式文件可以有返回值,也可以没 ...