[pixiv] https://www.pixiv.net/member_illust.php?mode=medium&illust_id=59001242

向大(hei)佬(e)势力学(di)习(tou)

问题描述:

在一个有m*n 个方格的棋盘中,每个方格中有一个正整数。现要从方格中取数,使任

意2 个数所在方格没有公共边,且取出的数的总和最大。试设计一个满足要求的取数算法。

«编程任务:

对于给定的方格棋盘,按照取数要求编程找出总和最大的数。

输入描述 Input Description

第1 行有2 个正整数m和n,分别表示棋盘的行数

和列数。接下来的m行,每行有n个正整数,表示棋盘方格中的数。

输出描述 Output Description

将取数的最大总和输出

样例输入 Sample Input

3 3

1 2 3

3 2 3

2 3 1

样例输出 Sample Output

11

数据范围及提示 Data Size & Hint

n,m<=30

最近学图论,精神大餐吃的太饱不消化,就多攻克一下网络流和二分图吧。

这道题其实是最小割的应用,兼有二分图的思想在里面。想想,要使选出来的最大,就要使留下来的最小,而我们刚好有求最小的东西(最大流=最小割)

方格是天然的二分图,可以将方格染成交叉的黑白图,黑的丢一边,白的丢一边,相邻的连边,容量正无穷,表示不能割掉。源点与黑的连边,容量即为格子中的数,白点与汇点连边,同理。割掉的边即为舍弃的格子。

代码(注意点的编号不能重复)

#include<cstdio>
#include<cstring>
#include<queue>
#include<algorithm>
using namespace std; const int oo=0x3f3f3f3f; int m,n,a[50][50],sz=0,top;
struct Node{
int to,nxt,flow,cap;
}r[6000];
int head[3000],hh=1,dep[3000];
queue<int> q; void adde(int fr,int to,int val){
hh++;
r[hh].to=to;
r[hh].flow=0;
r[hh].cap=val;
r[hh].nxt=head[fr];
head[fr]=hh;
hh++;
r[hh].to=fr;
r[hh].flow=0;
r[hh].cap=0;
r[hh].nxt=head[to];
head[to]=hh;
}
bool bfs(){
while(!q.empty()) q.pop();
memset(dep,-1,sizeof(dep));
q.push(0);
dep[0]=0;
while(!q.empty()){
int u=q.front();q.pop();
for(int i=head[u];i;i=r[i].nxt){
int v=r[i].to;
if(dep[v]==-1&&r[i].cap-r[i].flow>0){
dep[v]=dep[u]+1;
q.push(v);
}
}
if(u==top) return true;
}
return false;
}
int dfs(int u,int delta){
if(u==top||delta==0) return delta;
int rt=0,f;
for(int i=head[u];i;i=r[i].nxt){
int v=r[i].to;
if(dep[v]==dep[u]+1&&r[i].cap-r[i].flow>0){
f=dfs(v,min(delta,r[i].cap-r[i].flow));
rt+=f;
delta-=f;
r[i].flow+=f;
r[i^1].flow-=f;
}
if(delta==0) break;
}
return rt;
}
int main(){
top=2500;
int tot=0;
scanf("%d%d",&m,&n);
for(int i=1;i<=m;i++)
for(int j=1;j<=n;j++){
scanf("%d",&a[i][j]);
tot+=a[i][j];
if((i+j)%2==0){
adde(0,i*50+j,a[i][j]);
if(i-1>0) adde(i*50+j,(i-1)*50+j,oo);
if(i+1<=m) adde(i*50+j,(i+1)*50+j,oo);
if(j-1>0) adde(i*50+j,i*50+j-1,oo);
if(j+1<=n) adde(i*50+j,i*50+j+1,oo);
}
if((i+j)%2!=0){
adde(i*50+j,top,a[i][j]);
}
}
int ans=0;
while(bfs()){
ans+=dfs(0,oo);
}
printf("%d",tot-ans);
return 0;
}

总结:

1、遇到求最值问题,可从最大、最小两个方向出发。

2、转化、建边也是很需要思维的东西

3、天然的二分图:方格。当一个东西有两个元素时,可考虑二分图或网络流(源点汇点各一方嘛)

【codevs1907】【方格取数3】二分图最大带权独立集的更多相关文章

  1. HDU1569 方格取数(2) —— 二分图点带权最大独立集、最小割最大流

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

  2. codevs1907 方格取数 3

    «问题描述:在一个有m*n 个方格的棋盘中,每个方格中有一个正整数.现要从方格中取数,使任意2 个数所在方格没有公共边,且取出的数的总和最大.试设计一个满足要求的取数算法.«编程任务:对于给定的方格棋 ...

  3. luogu2774 方格取数问题 二分图最小权点覆盖集

    题目大意:在一个有 m*n 个方格的棋盘中,每个方格中有一个正整数.现要从方格中取数,使任意 2 个数所在方格没有公共边,输出这些数之和的最大值. 思路:这种各个点之间互相排斥求最大值的题,往往需要利 ...

  4. BZOJ 1475 方格取数(二分图最大点权独立集)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=1475 [题目大意] 给出一个n*n的方格,从中取一些不相邻的数字,使得和最大 [题解] ...

  5. 洛谷 - P2774 - 方格取数问题 - 二分图最大独立点集 - 最小割

    https://www.luogu.org/problemnew/show/P2774 把两个相邻的节点连边,这些边就是要方便最小割割断其他边存在的,容量无穷. 这种类似的问题的话,把二分图的一部分( ...

  6. 【codevs1907】方格取数3(最大流最小割定理)

    网址:http://codevs.cn/problem/1907/ 题意:在一个矩阵里选不相邻的若干个数,使这些数的和最大. 我们可以把它看成一个最小割,答案就是矩阵中的所有数-最小割.先把矩阵按国际 ...

  7. 线性规划与网络流24题●09方格取数问题&13星际转移问题

    ●(做codevs1908时,发现测试数据也涵盖了1907,想要一并做了,但因为“技术”不佳,搞了一上午) ●09方格取数问题(codevs1907  方格取数3) 想了半天,也没成功建好图: 无奈下 ...

  8. Luogu_2774 方格取数问题

    Luogu_2774 方格取数问题 二分图最小割 第一次做这种题,对于某些强烈暗示性的条件并没有理解到. 也就是每一立刻理解到是这个图是二分图. 为什么? 横纵坐标为奇数的只会和横纵坐标为偶数的相连. ...

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

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

随机推荐

  1. Jmeter获取Cookie并传递到下一个线程---跨线程后cookie找不到了

    网上找了一堆文章没有一个是实际操作的,自己边试边查边摸索终于找到了一个全套的办法. 原创文章,转载请说明出处. 1.取得cookie 直接这样写就可以了${COOKIE_JSESSIONID},当然具 ...

  2. jsonp、瀑布流布局、组合搜索、多级评论(评论树)、Tornado初识

    1.JSONP原理剖析以及实现 1.1 同源策略限制 用django分别建立两个项目,jsonp01和jsonp02,然后再在这两个项目里分别建立一个app,比如名字叫jsonp1.jsonp2:js ...

  3. Mysql忘记root密码怎么办?(已解决)

    为了写这篇文档,假装一下忘记密码!!!! 首先我数据库是正常的,可以使用. root@localhost:~# mysql -uroot -p mysql> mysql> show dat ...

  4. go语言的学习网站

    1)http://www.runoob.com/go/go-data-types.html 2)https://github.com/Unknwon/the-way-to-go_ZH_CN/blob/ ...

  5. Java TCP通信概念及实例

    TCP/UDP 协议 通俗解释: TCP协议和UDP协议的区别类似于电话系统和邮政系统. <1>TCP:类似于电话系统,建立双向的通信通道,确定连接,话音顺序接听. <2>UD ...

  6. c#中onclick事件请求的两种区别

    在C#中如果是asp控件的button有两个click的调用,一个是OnClick,一个是OnClientClick.那么这两者有什么区别呢,下面就来说说区别. <asp:Button ID=& ...

  7. 【PHP】- 启用CRUL和GD

    1.修改php.ini文件,确保extension_dir配置为php目录的ext,如图: 2.修改php.ini文件,将以下两项取消注释: 3.将php目录下的 libeay32.dll 和 lib ...

  8. java 图形化界面笔记(1)

    目录 JFrame窗体......................................................................................... ...

  9. [洛谷P4178]Tree

    题目大意:给一棵树,问有多少条路径长度小于等于$k$ 题解:点分治 卡点:无 C++ Code: #include <cstdio> #include <algorithm> ...

  10. 无法定位程序输入点GetTickCount64 在动态链接库kernel32.dll上

    winxp系统,在使用boost中的thread中的sleep的时候出现“无法定位程序输入点GetTickCount64 在动态链接库kernel32.dll上”的错误, 1.在引用boost库之前( ...