BZOJ1565: [NOI2009]植物大战僵尸

时间:2023-11-26 09:27:50

Description

BZOJ1565: [NOI2009]植物大战僵尸

Input

BZOJ1565: [NOI2009]植物大战僵尸

Output

仅包含一个整数,表示可以获得的最大能源收入。注意,你也可以选择不进行任何攻击,这样能源收入为0。

Sample Input

3 2
10 0
20 0
-10 0
-5 1 0 0
100 1 2 1
100 0

Sample Output

25

HINT

在样例中, 植物P1,1可以攻击位置(0,0), P2, 0可以攻击位置(2,1)。 
一个方案为,首先进攻P1,1, P0,1,此时可以攻击P0,0 。共得到能源收益为(-5)+20+10 = 25。注意, 位置(2,1)被植物P2,0保护,所以无法攻击第2行中的任何植物。 
【大致数据规模】
约20%的数据满足1 ≤ N, M ≤ 5;
约40%的数据满足1 ≤ N, M ≤ 10;
约100%的数据满足1 ≤ N ≤ 20,1 ≤ M ≤ 30,-10000 ≤ Score ≤ 10000 。

经典的最大权闭合子图,我们把依赖关系成环的节点以及依赖一个环的节点删掉,然后就是模板题。
然后我写了个Tarjan。
#include<cstdio>
#include<cctype>
#include<queue>
#include<cmath>
#include<cstring>
#include<algorithm>
#define rep(i,s,t) for(int i=s;i<=t;i++)
#define dwn(i,s,t) for(int i=s;i>=t;i--)
#define ren for(int i=first[x];i;i=next[i])
using namespace std;
const int BufferSize=<<;
char buffer[BufferSize],*head,*tail;
inline char Getchar() {
if(head==tail) {
int l=fread(buffer,,BufferSize,stdin);
tail=(head=buffer)+l;
}
return *head++;
}
inline int read() {
int x=,f=;char c=Getchar();
for(;!isdigit(c);c=Getchar()) if(c=='-') f=-;
for(;isdigit(c);c=Getchar()) x=x*+c-'';
return x*f;
}
const int maxn=;
const int maxm=;
const int inf=1e9;
struct ISAP{
struct tedge{int x,y,w,next;}adj[maxm];int ms,fch[maxn];
int d[maxn],s[maxn],cur[maxn],gap[maxn],n,top;
void init(int n){
this->n=n;ms=;top=;
memset(d,-,sizeof(d));
memset(fch,-,sizeof(fch));
return;
}
void AddEdge(int u,int v,int w){
adj[ms]=(tedge){u,v,w,fch[u]};fch[u]=ms++;
adj[ms]=(tedge){v,u,,fch[v]};fch[v]=ms++;
return;
}
void bfs(){
queue<int>Q;Q.push(n);d[n]=;
while(!Q.empty()){
int u=Q.front();Q.pop();
for(int i=fch[u];i!=-;i=adj[i].next){
int v=adj[i].y;
if(d[v]==-) d[v]=d[u]+,Q.push(v);
}
} return;
}
int solve(int S,int T){
n=T;bfs();int k=S,i,flow=;
for(i=;i<=n;i++) cur[i]=fch[i],gap[d[i]]++;
while(d[S]<n){
if(k==n){
int mi=inf,pos;
for(i=;i<top;i++) if(adj[s[i]].w<mi) mi=adj[s[i]].w,pos=i;
for(i=;i<top;i++) adj[s[i]].w-=mi,adj[s[i]^].w+=mi;
flow+=mi;top=pos;k=adj[s[top]].x;
}
for(i=cur[k];i!=-;i=adj[i].next){
int v=adj[i].y;
if(adj[i].w&&d[k]==d[v]+){cur[k]=i;k=v;s[top++]=i;break;}
}
if(i==-){
int lim=n;
for(i=fch[k];i!=-;i=adj[i].next){
int v=adj[i].y;
if(adj[i].w&&d[v]<lim) lim=d[v],cur[k]=i;
} if(--gap[d[k]]==) break;
d[k]=lim+;gap[d[k]]++;
if(k!=S) k=adj[s[--top]].x;
}
} return flow;
}
}sol;
int n,m,val[maxn];
int id(int x,int y) {return (x-)*m+y;}
int from[maxm],to[maxm],first[maxn],next[maxm],e;
void AddEdge(int u,int v) {
from[++e]=u;to[e]=v;next[e]=first[u];first[u]=e;
}
int S[maxn],scn[maxn],cnt[maxn],top,num,pre[maxn],low[maxn],mark[maxn],dfs_clock;
void dfs(int x) {
low[x]=pre[x]=++dfs_clock;S[++top]=x;
for(int i=first[x];i;i=next[i]) {
if(!pre[to[i]]) dfs(to[i]),low[x]=min(low[x],low[to[i]]);
else if(!scn[to[i]]) low[x]=min(low[x],pre[to[i]]);
}
if(low[x]==pre[x]) {
num++;for(;;) {
int u=S[top--];
scn[u]=num;cnt[num]++;
if(u==x) break;
}
}
}
int main() {
n=read();m=read();
rep(i,,n) rep(j,,m) {
val[id(i,j)]=read();
int k=read(),u=id(i,j);
while(k--) {
int x=read()+,y=read()+;
AddEdge(id(x,y),u);
}
if(j!=m) AddEdge(id(i,j),id(i,j+));
}
rep(i,,n*m) if(!pre[i]) dfs(i);
rep(i,,n*m) if(cnt[scn[i]]>) mark[i]=;
int S=n*m+,T=n*m+,sum=;sol.init(T);
rep(i,,e) if(mark[to[i]]) sol.AddEdge(from[i],T,1e9);
else sol.AddEdge(from[i],to[i],1e9);
rep(i,,n*m) if(!mark[i]) {
if(val[i]>) sum+=val[i],sol.AddEdge(S,i,val[i]);
else sol.AddEdge(i,T,-val[i]);
}
printf("%d\n",sum-sol.solve(S,T));
return ;
}