http://acm.hdu.edu.cn/showproblem.php?pid=1565

题意:

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

思路:

最大点权独立集=点权之和-最小点权覆盖集=最小割=最大流

先来看最小点权覆盖集,也就是选取点覆盖所有的边,并且权值要最小。

解决方法是:

从源点向X集连边,容量为点的权值,Y集向汇点连边,容量也为点的权值。如果u和v这两个点相连的话,则将这两个点连一条有向边,容量为INF,因为我们要割的不是这个。这样,从s到t的路径中,就包含了所有的边,最小点覆盖也就是连通所有边,最小割就是让所有边都不连通,于是求个最大流即可。

这样一来,最大点权独立集也就可以求出来了。

 #include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<queue>
using namespace std;
typedef long long LL; const int maxn=+;
const int INF=0x3f3f3f3f; struct Edge
{
int from,to,cap,flow;
Edge(int u,int v,int w,int f):from(u),to(v),cap(w),flow(f){}
}; struct Dinic
{
int n,m,s,t;
vector<Edge> edges;
vector<int> G[maxn];
bool vis[maxn];
int cur[maxn];
int d[maxn]; void init(int n)
{
this->n=n;
for(int i=;i<n;++i) G[i].clear();
edges.clear();
} void AddEdge(int from,int to,int cap)
{
edges.push_back( Edge(from,to,cap,) );
edges.push_back( Edge(to,from,,) );
m=edges.size();
G[from].push_back(m-);
G[to].push_back(m-);
} bool BFS()
{
queue<int> Q;
memset(vis,,sizeof(vis));
vis[s]=true;
d[s]=;
Q.push(s);
while(!Q.empty())
{
int x=Q.front(); Q.pop();
for(int i=;i<G[x].size();++i)
{
Edge& e=edges[G[x][i]];
if(!vis[e.to] && e.cap>e.flow)
{
vis[e.to]=true;
d[e.to]=d[x]+;
Q.push(e.to);
}
}
}
return vis[t];
} int DFS(int x,int a)
{
if(x==t || a==) return a;
int flow=, f;
for(int &i=cur[x];i<G[x].size();++i)
{
Edge &e=edges[G[x][i]];
if(d[e.to]==d[x]+ && (f=DFS(e.to,min(a,e.cap-e.flow) ) )>)
{
e.flow +=f;
edges[G[x][i]^].flow -=f;
flow +=f;
a -=f;
if(a==) break;
}
}
return flow;
} int Maxflow(int s,int t)
{
this->s=s; this->t=t;
int flow=;
while(BFS())
{
memset(cur,,sizeof(cur));
flow +=DFS(s,INF);
}
return flow;
}
}DC; int n;
int map[][];
int dx[]={,,,-};
int dy[]={,-,,}; int main()
{
while(~scanf("%d",&n))
{
int sum=;
int src=,dst=n*n+;
DC.init(dst+);
for(int i=;i<=n;i++)
for(int j=;j<=n;j++)
{
scanf("%d",&map[i][j]);
sum+=map[i][j];
}
for(int i=;i<=n;i++)
for(int j=;j<=n;j++)
{
int id=(i-)*n+j;
int t=(i+j)%;
if(t)
{
DC.AddEdge(src,id,map[i][j]);
for(int k=;k<;k++)
{
int x=dx[k]+i;
int y=dy[k]+j;
if(x<||x>n||y<||y>n) continue;
DC.AddEdge(id,(x-)*n+y,INF);
}
}
else DC.AddEdge(id,dst,map[i][j]);
}
int ans=DC.Maxflow(src,dst);
printf("%d\n",sum-ans);
}
return ;
}

最新文章

  1. Android开发学习—— 创建项目时,不是继承activity,而是继承ActionBarActivity
  2. [BI项目记]-搭建代码管理环境之签入代码
  3. Fragment的使用(一)
  4. SQL Network Interfaces, error: 50 - 发生了 Local Database Runtime 错误。无法创建自动实例。
  5. JavaScript - 获取高度
  6. 转!!Java中关于Null的9个解释(Java Null详解)
  7. 360 chrome 国际版能够隐藏用户保存的密码
  8. Project Euler:Problem 55 Lychrel numbers
  9. OC基础8:分类和协议
  10. C#--遍历目录实例
  11. DELPHI删除记录文件
  12. 002 网上看的unity学习路线
  13. CentOS7下解决yum install mysql-server 异常:No package mysql-server available.问题
  14. Python CGI编程
  15. 利用开源审计插件对mysql进行审计
  16. Webform中&lt;%%&gt;
  17. 关于ping地址的几个意义
  18. https连接器
  19. vue-cli 构建项目在IE中无法运行解决方式(build之后可运行)
  20. [CocoaPods]使用Gemfile

热门文章

  1. 最小树形图(hdu4009)
  2. js 格式验证大全
  3. pandas读取保存数据
  4. type=&quot;submit&quot;
  5. javascript关闭网页的几种方法
  6. Redundant Paths---poj3177(双连通分量)
  7. Tomcat安装(linux系统)
  8. 迅雷7 纯净版v7.9.18.4724
  9. raspberry pi 树莓派作为比特比矿机
  10. hdu1286(找新朋友)&amp;&amp;POJ2407Relatives(欧拉函数模版题)