[Luogu 2805] NOI2009 植物大战僵尸
<题目链接>
这题是个比较经典的最大权闭合子图,可以建图转化为最小割问题,再根据最大流最小割定理,采用任意一种最大流算法求得。
对于每个点,如果点权w为正,则从源点到这个点连一条边权为w的有向边;否则如果w为负则从这个点到汇点连一条有向边。加边的时候预处理出反图的每个点入度。
其次,每一个被保护的点到保护它的点连一条边权为INF的有向边。
注意同一行的左侧点受到右侧点的间接保护,因此对于每一个不位于当前行最右的点,都要向其右侧的一个点连一条INF有向边(连一条即可,不必连所有)。
初始化完成后,在反向图(惯用链式前向星的表示,在跑最小割之前,边权为0的边就是反向图的边)上拓扑排序,删去在环中的点,留下有效点。
然后在建的这个图上跑最小割。
最终的答案为:源点到「与源点相连的所有有效点」的边权和减去网络最小割。
建模是个好能力,希望包括我在内的每一个OIer都能通过不懈努力获得。
加油。
#include <algorithm>
#include <climits>
#include <cstdio>
#include <cstring>
#include <queue>
using namespace std;
const int MAXN=10000,MAXM=900000;
bool exist[MAXN]/*拓扑排序有效点标记*/,vis[MAXN];
int n,m,N,S,T,cnt,ans,head[MAXN],cur[MAXN]/*当前弧*/,inn[MAXN]/*反图入度*/,dis[MAXN]/*层次图*/;
struct edge
{
int nxt,to,w;
}e[MAXM];
void AddEdge(int x,int y,int w)
{
e[++cnt].nxt=head[x];
e[cnt].to=y;
e[cnt].w=w;
head[x]=cnt;
}
void AddEdges(int x,int y,int w)
{
AddEdge(x,y,w);
AddEdge(y,x,0);
++inn[x];//预处理反图入度
}
void T_Sort(int S)
{
queue<int> q;//栈或队列都可以(优先队列或者手写堆其实也没问题)
for(int i=S;i<=T;++i)
if(!inn[i])
q.push(i);//入度为0的入队
while(!q.empty())
{
int x=q.front();
q.pop();
exist[x]=1;
for(int i=head[x],t;i;i=e[i].nxt)
if(!e[i].w && !--inn[t=e[i].to])
q.push(t);
}
for(int i=head[S];i;i=e[i].nxt)
if(!inn[e[i].to])
ans+=e[i].w;//事先累加源点到「与源点相连的所有有效点」的边权
}
bool BFS(int S)
{
queue<int> q;
memset(dis,0,sizeof dis);
memset(vis,0,sizeof vis);
q.push(S);
vis[S]=1;
while(!q.empty())
{
int x=q.front();
q.pop();
for(int i=head[x],t;i;i=e[i].nxt)
if(e[i].w && exist[t=e[i].to] && !vis[t])
{
vis[t]=1;
dis[t]=dis[x]+1;
q.push(t);
}
if(vis[T])//已经处理到汇点就退出
return 1;
}
return 0;
}
int DFS(int x,int k)
{
if(x==T)
return k;
int tmp=0;//这个累加必须有,否则最后一个点会被卡掉。
for(int i=cur[x],t,f;i;i=e[i].nxt)
if(e[i].w && exist[t=e[i].to] && dis[t]==dis[x]+1 && (f=DFS(t,min(k,e[i].w))))
{
cur[x]=i;
e[i].w-=f;//当前边的边权
e[((i-1)^1)+1].w+=f;//反向边的边权
tmp+=f;
k-=f;
}
if(!tmp)
dis[x]=-1;//必须有的操作
return tmp;
}
void Dinic(void)
{
int f;
while(BFS(S))//预处理层次图,直到无法间出层次图
while(memcpy(cur,head,sizeof cur),f=DFS(S,INT_MAX))//跑DFS直到当前层次图上无法增广。
ans-=f;//连接源点的边权和减去最小割
}
int main(int argc,char *argv[])
{
scanf("%d %d",&n,&m);
N=n*m,T=N+1;//N是总点数,T是汇点
for(int i=1,w,k;i<=N;++i)
{
scanf("%d %d",&w,&k);
if(i%m)//如果不是当前行最后一个点,就向右连一条边。
AddEdges(i,i+1,INT_MAX);
if(w>0)
AddEdges(S,i,w);//连源点
else if(w<0)
AddEdges(i,T,-w);//连汇点
for(int j=1,r,c;j<=k;++j)
{
scanf("%d %d",&r,&c);
AddEdges(r*m+c+1,i,INT_MAX);//保护关系
}
}
T_Sort(S);//拓扑排序
Dinic();//最小割
printf("%d",ans);
return 0;
}
谢谢阅读。
[Luogu 2805] NOI2009 植物大战僵尸的更多相关文章
- 图论(网络流):COGS 410. [NOI2009] 植物大战僵尸
410. [NOI2009] 植物大战僵尸 ★★★ 输入文件:pvz.in 输出文件:pvz.out 简单对比时间限制:2 s 内存限制:512 MB [问题描述] Plants vs ...
- P2805 [NOI2009]植物大战僵尸
题目地址:P2805 [NOI2009]植物大战僵尸 最大权闭合子图 若有向图 \(G\) 的子图 \(V\) 满足: \(V\) 中顶点的所有出边均指向 \(V\) 内部的顶点,则称 \(V\) 是 ...
- COGS410. [NOI2009] 植物大战僵尸
410. [NOI2009] 植物大战僵尸 ★★★ 输入文件:pvz.in 输出文件:pvz.out 简单对比时间限制:2 s 内存限制:512 MB [问题描述] Plants vs ...
- BZOJ 1565: [NOI2009]植物大战僵尸
1565: [NOI2009]植物大战僵尸 Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 2317 Solved: 1071[Submit][Stat ...
- 【刷题】BZOJ 1565 [NOI2009]植物大战僵尸
Description Plants vs. Zombies(PVZ)是最近十分风靡的一款小游戏.Plants(植物)和Zombies(僵尸)是游戏的主角,其中Plants防守,而Zombies进攻. ...
- 【bzoj1565】[NOI2009]植物大战僵尸
1565: [NOI2009]植物大战僵尸 Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 2164 Solved: 1001[Submit][Stat ...
- 【最大权闭合子图 tarjan】bzoj1565: [NOI2009]植物大战僵尸
dinic+tarjan板子练手题 Description Plants vs. Zombies(PVZ)是最近十分风靡的一款小游戏.Plants(植物)和Zombies(僵尸)是游戏的主角,其 中P ...
- BZOJ1565: [NOI2009]植物大战僵尸
Description Input Output 仅包含一个整数,表示可以获得的最大能源收入.注意,你也可以选择不进行任何攻击,这样能源收入为0. Sample Input 3 2 10 0 20 0 ...
- 【bzoj1565】 NOI2009—植物大战僵尸
http://www.lydsy.com/JudgeOnline/problem.php?id=1565 (题目链接) 题意 给出$n*m$的棋盘,僵尸攻击每个格子可以获得$v$的分数,每个格子又会保 ...
随机推荐
- Java中抽象类也能实例化
在Java中抽象类真的不能实例化么? 在学习的过程中,发现了一个问题,抽象类在没有实现所有的抽象方法前是不可以通过new来构建该对象的,但是抽象方法却是可以有自己的构造方法的.这样就把我搞糊涂了,既然 ...
- 【Linux】- apt-get命令
apt-get,是一条linux命令,适用于deb包管理式的操作系统,主要用于自动从互联网的软件仓库中搜索.安装.升级.卸载软件或操作系统. Advanced Package Tool,又名apt-g ...
- 爬虫之手机APP抓包教程-亲测HTTP和HTTPS均可实现
当下很多网站都有做自己的APP端产品,一个优秀的爬虫工程师,必须能够绕过难爬取点而取捷径,这是皆大欢喜的.但是在网上收罗和查阅了无数文档和资料,本人亲测无数次,均不能正常获取HTTPS数据,究其原因是 ...
- 再看RCU
从昨天晚上开始,我就立志要把RCU拿下,昨晚加今天早上看了RCU的东西,太细节的原理我就不扣了,放弃,知道RCU是怎么用的就可以了,赶紧看文件系统中dcache的管理了. 说说RCU,RCU其实是很简 ...
- ASP.NET MVC下使用文件上传和IIS7下的默认设置限制了上传大小的方法
不多说了,直接用别人的 http://www.cnblogs.com/jiekzou/p/4491505.html
- BZOJ4866 Ynoi2017由乃的商场之旅(莫队)
显然能重排为回文串相当于出现次数为奇数的字母不超过一个.考虑莫队,问题在于如何统计添加/删除一位的贡献.将各字母出现次数奇偶性看做二进制数,做一个前缀和一个后缀和.在右端添加一位时,更新区间的前缀.后 ...
- 【题解】CQOI2012局部最小值
上课讲的一道题,感觉也挺厉害的~正解是容斥 + 状压dp.首先我们容易发现一共可能的局部最小值数量是十分有限的,最多也只有 \(8\) 个.所以我们可以考虑状压. 建立出状态 \(f[i][j]\) ...
- 【题解】NOIP2015推销员
……普及组的题目都做不出来……(:´д`)ゞ……再这样下去要退役了啊…… 不过不管怎样感觉这题还是蛮好的,也要记录一下下~ 我们注意到数据的范围,n 是 1e5, 又有 1e5组询问,暴力大概是 \( ...
- CF633C:Spy Syndrome 2——题解
https://vjudge.net/problem/CodeForces-633C http://codeforces.com/problemset/problem/633/C 点击这里看巨佬题解 ...
- HDOJ(HDU).1035 Robot Motion (DFS)
HDOJ(HDU).1035 Robot Motion [从零开始DFS(4)] 点我挑战题目 从零开始DFS HDOJ.1342 Lotto [从零开始DFS(0)] - DFS思想与框架/双重DF ...