1.飞行员配对方案问题
二分图最大匹配。
#includeusing namespace std;inline int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}#define MN 2005#define S 0#define T 205#define inf 200000000struct edge{ int to,w,nex;}e[MN*T];int hr[T+5],cnt=1,d[T+5];int q[T+5],top,ans;int m,n;void ins(int f,int t,int w){ e[++cnt]=edge{t,w,hr[f]};hr[f]=cnt; e[++cnt]=edge{f,0,hr[t]};hr[t]=cnt;}bool bfs(){ memset(d,0,sizeof d); register int i,j; for(d[q[top=i=1]=S]=1;i<=top;++i) for(j=hr[q[i]];j;j=e[j].nex) if(e[j].w&&!d[e[j].to]) d[q[++top]=e[j].to]=d[q[i]]+1; return d[T];}int dfs(int x,int f){ if(x==T) return f; register int i,used=0; for(i=hr[x];i;i=e[i].nex) if(e[i].w&&d[e[i].to]==d[x]+1){ int w=dfs(e[i].to,min(f-used,e[i].w)); used+=w;e[i].w-=w;e[i^1].w+=w; if(f==used) return f; } return d[x]=-1,used;}int main(){ n=read(),m=read(); register int x,y; while(1){ x=read(),y=read(); if(x==-1&&y==-1) break; ins(x,y+100,1); } for(x=1;x<=n;x++) ins(S,x,1); for(y=1;y<=m;y++) ins(y+100,T,1); while(bfs()) ans+=dfs(S,inf); printf("%d\n",ans); if(!ans) printf("No Solution!\n"); for(int i=hr[S];i;i=e[i].nex)if(!e[i].w){ for(int j=hr[e[i].to];j;j=e[j].nex)if(!e[j].w&&e[j].to!=S) printf("%d %d\n",e[i].to,e[j].to-100); } return 0;}
2.太空飞行计划问题
最大权闭合图?最小割
#includeusing namespace std;inline int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}#define MN 55int val[MN],sum;#define S 0#define T 105#define inf 200000000struct edge{ int to,w,nex;}e[MN*T];int hr[T+5],cnt=1,d[T+5];int q[T+5],top,ans;int m,n;void ins(int f,int t,int w){ e[++cnt]=(edge){t,w,hr[f]};hr[f]=cnt; e[++cnt]=(edge){f,0,hr[t]};hr[t]=cnt;}bool bfs(){ memset(d,0,sizeof d); register int i,j; for(d[q[top=i=1]=S]=1;i<=top;++i) for(j=hr[q[i]];j;j=e[j].nex) if(e[j].w&&!d[e[j].to]) d[q[++top]=e[j].to]=d[q[i]]+1; return d[T];}int dfs(int x,int f){ if(x==T) return f; int used=0; for(int i=hr[x];i;i=e[i].nex) if(e[i].w&&d[e[i].to]==d[x]+1){ int w=dfs(e[i].to,min(f-used,e[i].w)); used+=w; e[i].w-=w; e[i^1].w+=w; if(f==used) return f; } return d[x]=-1,used;}int main(){ n=read(),m=read(); register int i,j,ulen,num;char c[10000]; for(i=1;i<=n;i++){ val[i]=read(); ins(S,i,val[i]);sum+=val[i]; memset(c,0,sizeof c); cin.getline(c,10000); ulen=0; while(sscanf(c+ulen,"%d",&num)==1){ ins(i,50+num,inf); if(num==0) ulen++; else while(num) num/=10,ulen++; ulen++; } } for(i=1;i<=m;i++) ins(i+50,T,read()); while(bfs()) ans+=dfs(S,inf); for(i=1;i<=n;i++) if(d[i]>0) printf("%d ",i);puts(""); for(i=1;i<=m;i++) if(d[i+50]>0) printf("%d ",i);puts(""); printf("%d\n",sum-ans); return 0;}
3.最小路径覆盖问题
先拆点,最小路径覆盖=总点数-二分图的最大匹配。
#includeusing namespace std;inline int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}#define MN 155#define ME 15005#define inf 2000000000#define S 0#define T 305int n,m,ans;struct edge{ int to,w,nex;}e[ME*2];int cnt=1,hr[T+5],d[T+5],q[T+5],top;inline void ins(int f,int t,int w){ e[++cnt]=(edge){t,w,hr[f]};hr[f]=cnt; e[++cnt]=(edge){f,0,hr[t]};hr[t]=cnt;}bool bfs(){ memset(d,0,sizeof d); register int i,j; for(d[q[top=i=1]=S]=1;i<=top;i++) for(j=hr[q[i]];j;j=e[j].nex) if(e[j].w&&!d[e[j].to]) d[q[++top]=e[j].to]=d[q[i]]+1; return d[T];}int dfs(int x,int f){ if(x==T) return f;int used=0; for(int i=hr[x];i;i=e[i].nex) if(e[i].w&&d[e[i].to]==d[x]+1){ int w=dfs(e[i].to,min(e[i].w,f-used)); used+=w;e[i].w-=w;e[i^1].w+=w; if(used==f) return used; } return d[x]=-1,used;}bool prin[MN];inline void pr(int x){printf("%d ",x);prin[x]=1;}void pri(int x){ int i; for(i=hr[x];i;i=e[i].nex)if(e[i].to!=S&&!e[i].w){ pr(e[i].to-150);pri(e[i].to-150); }}int main(){ n=read(),m=read(); int u,v; for(int i=1;i<=n;i++) ins(S,i,1),ins(i+150,T,1); for(int i=1;i<=m;i++){ u=read(),v=read(); ins(u,v+150,1); } while(bfs()) ans+=dfs(S,inf); for(int i=1;i<=n;i++)if(!prin[i]){ pr(i);pri(i); puts(""); } printf("%d\n",n-ans); return 0;}
4.魔术球问题
仍然是最小路径覆盖问题,我们枚举答案,每次都新增一个点,在残量网络上跑最大流,知道最小路径覆盖超过n。
#includeusing namespace std;inline int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}#define Mans 1600#define inf 2000000000#define S 0#define T Mans*2struct edge{ int to,w,nex;}e[Mans*Mans*2];int cnt=1,hr[T+5],d[T+5],q[T+5],top;int cur[T+5];inline void ins(int f,int t,int w){ e[++cnt]=(edge){t,w,hr[f]};hr[f]=cnt; e[++cnt]=(edge){f,0,hr[t]};hr[t]=cnt;}bool bfs(){ memset(d,0,sizeof d); register int i,j; for(d[q[top=i=1]=S]=1;i<=top;i++) for(j=hr[q[i]];j;j=e[j].nex) if(e[j].w&&!d[e[j].to]) d[q[++top]=e[j].to]=d[q[i]]+1; return d[T];}int dfs(int x,int f){ if(x==T) return f;int used=0; for(int i=cur[x];i;i=e[i].nex){ cur[x]=i; if(e[i].w&&d[e[i].to]==d[x]+1){ int w=dfs(e[i].to,min(e[i].w,f-used)); used+=w;e[i].w-=w;e[i^1].w+=w; if(used==f) return used; } } return d[x]=-1,used;}int n,ans;bool insed[Mans+5];bool is(int x){ return sqrt(x)*sqrt(x)==x;}void insw(int x){ insed[x]=1; ins(S,x,1);ins(x+Mans,T,1); for(int i=sqrt(x)+1;i*i<2*x;++i) ins(i*i-x,x+Mans,1);}void reins(int x){ cnt=1;memset(hr,0,sizeof hr); register int i,j; for(i=1;i<=x;i++)for(int j=sqrt(i)+1;j*j<2*i;++j) ins(j*j-i,i+Mans,1); for(i=1;i<=x;i++) ins(S,i,1),ins(i+Mans,T,1);}bool prin[Mans+5];inline void pr(int x){printf("%d ",x);prin[x]=1;}void pri(int x){ register int i; for(i=hr[x];i;i=e[i].nex)if(e[i].to!=S&&!e[i].w){ pr(e[i].to-Mans);pri(e[i].to-Mans); }}int main(){ n=read(); register int i=1,j; for(j=1;j<=n;j++) for(;i<=Mans;i++){ if(!insed[i]) insw(i); while(bfs()){ memcpy(cur,hr,sizeof(int)*(T+5)); ans+=dfs(S,inf); } if(i-ans>j) break; } printf("%d\n",i-1); reins(i-1); while(bfs()){ memcpy(cur,hr,sizeof(int)*(T+5)); ans+=dfs(S,inf); } for(j=1;j<=i-1;j++)if(!prin[j]){ pr(j);pri(j); puts(""); } return 0;}
5.圆桌问题
还是匹配问题,源点向单位连容量为人数的边,桌子向汇点连容量为桌子大小的边就行了。
#includeusing namespace std;inline int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}#define MM 150#define MN 270#define ME (MM+1)*(MN+1)#define inf 2000000000#define S 0#define T 425int n,m,ans;struct edge{ int to,w,nex;}e[ME*2];int cnt=1,hr[T+5],d[T+5],q[T+5],cur[T+5],top;inline void ins(int f,int t,int w){ e[++cnt]=(edge){t,w,hr[f]};hr[f]=cnt; e[++cnt]=(edge){f,0,hr[t]};hr[t]=cnt;}bool bfs(){ memset(d,0,sizeof d); register int i,j; for(d[q[top=i=1]=S]=1;i<=top;i++) for(j=hr[q[i]];j;j=e[j].nex) if(e[j].w&&!d[e[j].to]) d[q[++top]=e[j].to]=d[q[i]]+1; return d[T];}int dfs(int x,int f){ if(x==T) return f;int used=0; for(int i=cur[x];i;i=e[i].nex){ cur[x]=i; if(e[i].w&&d[e[i].to]==d[x]+1){ int w=dfs(e[i].to,min(e[i].w,f-used)); used+=w;e[i].w-=w;e[i^1].w+=w; if(used==f) return used; } } return d[x]=-1,used;}int sm,x;int main(){ m=read(),n=read(); for(int i=1;i<=m;i++) x=read(),sm+=x,ins(S,i,x); for(int j=1;j<=n;j++) x=read(),ins(j+150,T,x); for(int i=1;i<=m;i++)for(int j=1;j<=n;j++) ins(i,j+150,1); while(bfs()){ memcpy(cur,hr,sizeof(int)*(T+5)); ans+=dfs(S,inf); } if(ans
6.最长不下降子序列问题
首先第1问平方dp就行了。dp[i]表示以i结尾的最长不下降子序列的长度。
第2问的话,要使用上一问的dp值,对图进行分层,每层的数向下一层且在自己之后比自己大的数连边连边,然后跑最大流。
第3问,把S到1号点和n号点到T的容量改为inf,再跑最大流。
#includeusing namespace std;inline int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}#define MN 505#define inf2 0x3f#define inf 10000000#define ME 250000#define S 0#define T 1005int cnt=1,d[T+5],hr[T+5],cur[T+5],q[T+5],top,ans;struct edge{ int to,w,nex;}e[ME];inline void ins(int f,int t,int w){ e[++cnt]=(edge){t,w,hr[f]};hr[f]=cnt; e[++cnt]=(edge){f,0,hr[t]};hr[t]=cnt;}inline bool bfs(){ memset(d,0,sizeof d); register int i,j; for(d[q[i=top=1]=S]=1;i<=top;i++) for(j=hr[q[i]];j;j=e[j].nex) if(e[j].w&&!d[e[j].to]) d[q[++top]=e[j].to]=d[q[i]]+1; return d[T];}inline int dfs(int x,int f){ if(x==T) return f;int used=0; for(int i=cur[x];i;i=e[i].nex){ cur[x]=i; if(e[i].w&&d[e[i].to]==d[x]+1){ int w=dfs(e[i].to,min(f-used,e[i].w)); e[i].w-=w,used+=w,e[i^1].w+=w; if(used==f) return used; } } return d[x]=-1,used;}inline void dinic(){ while(bfs()){ memcpy(cur,hr,sizeof(int)*(T+5)); ans+=dfs(S,inf); }}int n,a[MN],f[MN],ct;int main(){ n=read(); register int i,j; for(i=1;i<=n;i++)a[i]=read(); for(i=1;i<=n;i++)for(j=1;j <=a[i]) f[i]=max(f[i],f[j]+1); for(i=1;i<=n;i++) ct=max(ct,f[i]); printf("%d\n",ct+1); for(i=1;i<=n;i++){ if(f[i]==0) ins(S,i,1); ins(i,i+500,1); if(f[i]==ct) ins(i+500,T,1); for(j=i+1;j<=n;j++)if(a[j]>=a[i]&&f[j]-f[i]==1) ins(i+500,j,1); } dinic(); printf("%d\n",ans); ins(S,1,inf);ins(1,501,inf); if(f[n]==ct) ins(n,n+500,inf),ins(n+500,T,inf); dinic(); printf("%d\n",ans); return 0;}
7.试题库问题
还是匹配问题。
#includeusing namespace std;inline int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}#define MN 1000#define T 1025#define S 0#define inf 200000000struct edge{ int to,w,nex;}e[MN*T];int hr[T+5],cnt=1,d[T+5];int q[T+5],top,ans;int k,n;void ins(int f,int t,int w){ e[++cnt]=edge{t,w,hr[f]};hr[f]=cnt; e[++cnt]=edge{f,0,hr[t]};hr[t]=cnt;}bool bfs(){ memset(d,0,sizeof d); register int i,j; for(d[q[top=i=1]=S]=1;i<=top;++i) for(j=hr[q[i]];j;j=e[j].nex) if(e[j].w&&!d[e[j].to]) d[q[++top]=e[j].to]=d[q[i]]+1; return d[T];}int dfs(int x,int f){ if(x==T) return f; register int i,used=0; for(i=hr[x];i;i=e[i].nex) if(e[i].w&&d[e[i].to]==d[x]+1){ int w=dfs(e[i].to,min(f-used,e[i].w)); used+=w;e[i].w-=w;e[i^1].w+=w; if(f==used) return f; } return d[x]=-1,used;}int main(){ k=read(),n=read(); int sm=0,a; for(int i=1;i<=k;i++) a=read(),sm+=a,ins(1000+i,T,a); for(int i=1;i<=n;i++){ ins(S,i,1);a=read(); for(int j=1;j<=a;j++) ins(i,read()+1000,1); } while(bfs()) ans+=dfs(S,inf); if(ans==sm){ for(int i=1;i<=k;i++){ printf("%d:",i); for(int j=hr[1000+i];j;j=e[j].nex) if(e[j].to!=T&&e[j].w) printf(" %d",e[j].to); puts(""); } } else puts("No Solution!"); return 0;}
8.机器人路径规划问题
9.方格取数问题
二分图点权最大独立集。其实就是最小割,先给图黑白染色,分别连向S和T,相邻格子之间连inf的边,格子和S(或T)之间边的容量为格子内正整数的值。
#includeusing namespace std;inline int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}#define MN 105#define T 10005#define S 0#define inf 2e9struct edge{ int to,w,nex;}e[MN*T];int cnt=1,hr[T+5];inline void ins(int f,int t,int w){ e[++cnt]=(edge){t,w,hr[f]};hr[f]=cnt; e[++cnt]=(edge){f,0,hr[t]};hr[t]=cnt;}int d[T+5],q[T+5],ans,top;bool bfs(){ memset(d,0,sizeof d);register int i,j; for(d[q[i=top=1]=S]=1;i<=top;++i) for(j=hr[q[i]];j;j=e[j].nex) if(e[j].w&&!d[e[j].to]) d[q[++top]=e[j].to]=d[q[i]]+1; return d[T];}int dfs(int x,int f){ if(x==T) return f; int used=0; for(int i=hr[x];i;i=e[i].nex) if(e[i].w&&d[e[i].to]==d[x]+1){ int w=dfs(e[i].to,min(f-used,e[i].w)); used+=w;e[i].w-=w;e[i^1].w+=w; if(used==f) return used; } return d[x]=-1,used;}int n,m,sum;int g(int i,int j){ return m*(i-1)+j;}int main(){ n=read(),m=read(); register int i,j,x; for(i=1;i<=n;i++)for(j=1;j<=m;j++) x=read(),sum+=x,(i+j)&1?ins(S,g(i,j),x):ins(g(i,j),T,x); for(i=1;i<=n;i++)for(j=1;j<=m;j++) if((i+j)&1){ if(i>1) ins(g(i,j),g(i-1,j),inf); if(i 1) ins(g(i,j),g(i,j-1),inf); if(j
10.餐巾计划问题
最小费用最大流
把每天拆成x[i]和y[i],S向每个x[i]连容量inf,费用为p(餐巾价钱)的边,向y[i]连容量为need[i],费用为0的边,y[i]向y[i+1]连容量inf,费用为0的边,表示当日用剩的餐巾可以留到以后去处理,y[i]向x[i+m]连容量为inf,费用为f的边,表示送到快洗部,慢洗部同理。最后,x[i]向T连边容量为need[i],费用为0,这样用来控制最大流合法。
#includeusing namespace std;inline int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}#define inf 0x3f3f3f3f#define ME 100005#define MN 5005#define T 4005#define S 0long long mincost,maxflow;struct edge{ int to,w,c,nex;}e[ME];int hr[MN],cnt=1;inline void ins(int f,int t,int w,int c){ e[++cnt]=(edge){t,w,c,hr[f]};hr[f]=cnt; e[++cnt]=(edge){f,0,-c,hr[t]};hr[t]=cnt; }int d[MN],q[ME],l,r;bool inq[MN],vis[MN];bool spfa(){ memset(d,0x3f,sizeof d); q[l=r=MN]=T;d[T]=0;inq[T]=1; while(l<=r){ int u=q[l++];inq[u]=0; for(int i=hr[u];i;i=e[i].nex) if(e[i^1].w&&d[e[i].to]>d[u]-e[i].c){ d[e[i].to]=d[u]-e[i].c; if(!inq[e[i].to]) d[e[i].to]
11.航空路线问题
最大费用最大流。
题目就是要求找出两条不相交的从1号城市到n号城市的路径,考虑每个城市拆点,连边容量为1,费用为1,西边城市向东边城市连边,S向1号城市连容量为2,费用为0的边,然后跑最大费用最大流,和最小费用流的做法相似。
#includeusing namespace std;inline int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}#define inf 0x3f3f3f3f#define MN 105#define T 205#define S 0#define ME 90000int maxflow,mincost;struct edge{ int to,w,c,nex;}e[ME];int hr[T+5],cnt=1;inline void ins(int f,int t,int w,int c){ e[++cnt]=(edge){t,w,c,hr[f]};hr[f]=cnt; e[++cnt]=(edge){f,0,-c,hr[t]};hr[t]=cnt; }int d[T+5],q[ME*2+5],l,r;bool inq[T+5],vis[T+5];bool spfa(){ memset(d,128,sizeof d); q[l=r=ME]=T;d[T]=0;inq[T]=1; while(l<=r){ int u=q[l++];inq[u]=0; for(int i=hr[u];i;i=e[i].nex) if(e[i^1].w&&d[e[i].to] d[q[l]]?q[--l]=e[i].to:q[++r]=e[i].to,inq[e[i].to]=1; } } return d[S]>0;}int flow(int x,int f){ vis[x]=1; if(x==T) return f; int used=0,w; for(int i=hr[x];i;i=e[i].nex) if(!vis[e[i].to]&&d[x]-e[i].c==d[e[i].to]&&e[i].w){ w=flow(e[i].to,min(f-used,e[i].w)); used+=w;mincost+=w*e[i].c; e[i].w-=w,e[i^1].w+=w; if(f==used) return f; } return used;}inline void solve(){ while(spfa()){ do{ memset(vis,0,sizeof vis); maxflow+=flow(S,inf); }while(vis[T]); }}int n,m;char city[MN][20];char st[20],en[20];int sta,end;int ct=0,print[3][MN];inline void solve_(int x){ ct++; if(x==n) return; print[ct][1]=x; print[ct][0]=1;int now=x; while(now!=n){ for(int i=hr[now+100];i;i=e[i].nex){ if(e[i].to==n&&!e[i].w) return; if(!e[i].w&&e[i].to<100){ now=print[ct][++print[ct][0]]=e[i].to; break; } } }}bool pd(char a[],char b[]){ int len; if((len=strlen(a))!=strlen(b)) return false; for(int i=0;i end) swap(sta,end); ins(sta+100,end,1+(sta==1&&end==n),1); } for(i=2;i >1); printf("%s\n",city[1]); for(i=1;i<=print[2][0];i++) printf("%s\n",city[print[2][i]]); printf("%s\n",city[n]); for(i=print[1][0];i>=1;i--) printf("%s\n",city[print[1][i]]); printf("%s\n",city[1]); return 0;}
12.软件补丁问题
这是一道最短路径问题,首先,n<=20,状态压缩。然后根据每个补丁的需要,胡乱转移,spfa跑最短路就行了。
#includeusing namespace std;inline int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}#define MN 2005#define inf 0x3f3f3f3f#define INF dp[(1<<20)+5]int n,m,t[MN],f1[MN],f2[MN],d1[MN],d2[MN];char c1[MN],c2[MN];queue q;long long dp[(1<<20)+10];bool in[1<<20];void init(int x){ register int i; for(i=0;i f1[y]&&Cx>f2[y] for(int i=0;i >i)&1&&~(x>>i)&1) return false; if((f2[y]>>i)&1&&(x>>i)&1) return false; } return true;}int change(int x,int y,int z){ for(int i=0;i >i)&1&& (x>>i)&1) x-=1< >i)&1&&~(x>>i)&1) x+=1< dp[tp]+t[i]){ if(!in[xf])q.push(xf),in[xf]=1;dp[xf]=dp[tp]+t[i]; } } }}int main(){ n=read(),m=read(); register int i,j; for(i=1;i<=m;i++){ scanf("%d%s%s",&t[i],c1,c2); init(i); } bfs(); printf("%d\n",dp[0]==INF?0:dp[0]); return 0;}
13.星际转移问题([CTSC1999]家园)
首先,先用并查集判断一下起点和终点是否相连,然后开始枚举答案,每次t++就增加一组状态点,及飞船i在t时刻的状态,然后从t-1的状态点向t的状态连边,容量为飞船船的载客量,如果当前飞船在0号点,则S向它直接连一条容量为载客量的边,如果在终点,则直接向T连边。
#includeusing namespace std;inline int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}#define S 0#define MN 105#define ME 1000005#define inf 0x3f3f3f3f#define T ME-2struct edge{ int to,w,nex;}e[ME];int cnt=1,hr[T+5],cur[T+5];int d[T+5],q[T+5],top,ans;int fa[MN],val[MN];int getf(int x){ return x==fa[x]?fa[x]:fa[x]=getf(fa[x]);}void union_(int x,int y){ x=getf(x),y=getf(y); if(x==y) return; fa[x]=y;return;}inline void ins(int f,int t,int w){ e[++cnt]=(edge){t,w,hr[f]};hr[f]=cnt; e[++cnt]=(edge){f,0,hr[t]};hr[t]=cnt;}bool bfs(){ memset(d,0,sizeof d); register int i,j; for(d[q[top=i=1]=S]=1;i<=top;i++) for(j=hr[q[i]];j;j=e[j].nex) if(e[j].w&&!d[e[j].to]) d[q[++top]=e[j].to]=d[q[i]]+1; return d[T];}int dfs(int x,int f){ if(x==T) return f;int used=0; for(int i=cur[x];i;i=e[i].nex){ cur[x]=i; if(e[i].w&&d[e[i].to]==d[x]+1){ int w=dfs(e[i].to,min(e[i].w,f-used)); used+=w;e[i].w-=w;e[i^1].w+=w; if(used==f) return used; } } return d[x]=-1,used;}void dinic(){ while(bfs()){ memcpy(cur,hr,sizeof(int)*(T+5)); ans+=dfs(S,inf); }}int n,m,k,p[MN],v[MN][MN];int g(int x,int y){ if(x==0) return y; return x;}inline int solve(){ register int j,i,x,y; for(j=1;;++j){ ins(S,(j-1)*(n+1)+n+1,inf); for(i=1;i<=m;++i){ x=(j-1)%val[i],y=j%val[i]; if(v[i][x]==n+2) x=T; else x=(j-1)*(n+1)+v[i][x]; if(v[i][y]==n+2) y=T; else y=j*(n+1)+v[i][y]; ins(x,y,p[i]); } dinic(); if(ans>=k) return j; for(i=1;i<=n+1;++i) ins((j-1)*(n+1)+i,j*(n+1)+i,inf); }}int main(){ n=read();m=read();k=read(); register int i,j; for(i=1;i<=n+2;i++) fa[i]=i; for(i=1;i<=m;++i){ p[i]=read(),val[i]=read(); for(j=0;j
14.孤岛营救问题
最短路问题~f[i][j][k]表示到达(i,j)时手中的钥匙状态为k的最小步数,然后随便转移就行了,实现的话直接广搜会更方便。
#includeusing namespace std;inline int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}#define MN 15int n,m,p,K;int border[MN][MN][MN][MN],zz[MN][MN];bool vis[MN][MN][1<<10+5];struct zt{ int x,y,z,st;};const int xx[4]={ 0,1,0,-1},yy[4]={ 1,0,-1,0};queue q;int main(){ n=read(),m=read(),p=read();K=read(); register int i,j; int a,b,c,d,key,X,Y,step,t; for(i=1;i<=K;i++){ a=read(),b=read(),c=read(),d=read(),key=read(); if(key!=0) border[a][b][c][d]|=1< < n||Y>m) continue; t=border[tp.x][tp.y][X][Y]; if((tp.z&t)!=t) continue; if(vis[X][Y][tp.z|zz[X][Y]]) continue; vis[X][Y][tp.z|zz[X][Y]]=1; q.push((zt){X,Y,tp.z|zz[X][Y],step+1}); } } printf("-1\n"); return 0;}
15.汽车加油行驶问题
依旧时最短路~f[i][j][k]表示到达(i,j)时油量为k的最小步数。
#includeusing namespace std;inline int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}const int xx[4]={-1,0,1,0},yy[4]={ 0,1,0,-1};int n,k,a,b,c,ans=99999999;bool mp[105][105],inq[105][105][15];int f[105][105][14];struct nd{ int x,y,z;};queue q;//#define ju [tp.x][tp.y][tp.z]//#define uj [tp.x][tp.y][k]int main(){ n=read(),k=read(),a=read(),b=read(),c=read(); register int i,j; for(i=1;i<=n;i++)for(j=1;j<=n;j++)mp[i][j]=(read()==1); memset(f,127,sizeof f); f[1][1][k]=0;inq[1][1][k]=1;q.push((nd){ 1,1,k}); while(!q.empty()){ nd tp=q.front();q.pop();inq[tp.x][tp.y][tp.z]=0; if(mp[tp.x][tp.y]&&tp.z!=k){ if(f[tp.x][tp.y][k]>f[tp.x][tp.y][tp.z]+a){ f[tp.x][tp.y][k]=f[tp.x][tp.y][tp.z]+a; if(!inq[tp.x][tp.y][k])inq[tp.x][tp.y][k]=1,q.push((nd){tp.x,tp.y,k}); } continue; } if(f[tp.x][tp.y][k]>f[tp.x][tp.y][tp.z]+a+c){ f[tp.x][tp.y][k]=f[tp.x][tp.y][tp.z]+a+c; if(!inq[tp.x][tp.y][k]) inq[tp.x][tp.y][k]=1,q.push((nd){tp.x,tp.y,k}); } if(tp.z>0)for(i=0;i<4;++i){ int nx=tp.x+xx[i],ny=tp.y+yy[i]; if(nx<1||nx>n||ny<1||ny>n) continue; int ins=(xx[i]<0||yy[i]<0)?b:0; if(f[nx][ny][tp.z-1]>f[tp.x][tp.y][tp.z]+ins){ f[nx][ny][tp.z-1]=f[tp.x][tp.y][tp.z]+ins; if(!inq[nx][ny][tp.z-1]) inq[nx][ny][tp.z-1]=1,q.push((nd){nx,ny,tp.z-1}); } } } for(i=0;i<=k;++i) ans=min(ans,f[n][n][i]); printf("%d\n",ans);}
16.数字梯形问题
拆点,控制点的流量,连边,控制边的流量,同时费用为1,跑最大费用最大流,OK。
#includeusing namespace std;inline int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}#define MN 45#define ME 30000 #define S 0#define T 1005#define T1 1004#define inf 0x3f3f3f3fint mincost,maxflow;struct edge{ int to,w,c,nex;}e[ME+5];int cnt=1,l,u,r,d[T+5],inq[T+5],vis[T+5],q[ME*2+5],hr[T+5];inline void ins(int f,int t,int w,int c){ e[++cnt]=(edge){t,w,c,hr[f]};hr[f]=cnt; e[++cnt]=(edge){f,0,-c,hr[t]};hr[t]=cnt;}inline bool spfa(){ memset(d,128,sizeof d); q[l=r=ME+1]=T;inq[T]=1;d[T]=0; while(l<=r){ int u=q[l++];inq[u]=0; for(int i=hr[u];i;i=e[i].nex) if(e[i^1].w&&d[e[i].to] d[q[l]]?q[--l]=e[i].to:q[++r]=e[i].to,inq[e[i].to]=1; } } return d[S]>d[1003];}inline int mcf(int x,int f){ vis[x]=1; if(x==T) return f; int used=0,w; for(int i=hr[x];i;i=e[i].nex) if(!vis[e[i].to]&&d[e[i].to]==d[x]-e[i].c&&e[i].w){ w=mcf(e[i].to,min(f-used,e[i].w)); used+=w;e[i].w-=w;e[i^1].w+=w; mincost+=e[i].c*w; if(f==used) return used; } return used;}inline void solve(){ while(spfa()){ // for(int i=0;i<=20;i++) printf("%d ",d[i]);puts(""); do{ memset(vis,0,sizeof vis); maxflow+=mcf(S,inf); }while(vis[T]); }}int n,m,num[MN][MN],mun[MN][MN],cc;int main(){ m=read(),n=read(); register int i,j; for(i=1;i<=n;++i)for(j=1;j<=m+i-1;++j) num[i][j]=++cc; for(i=1;i<=n;++i)for(j=1;j<=m+i-1;++j) mun[i][j]=read(); for(i=1;i<=m;++i) ins(S,num[1][i],1,0); for(i=1;i<=n;++i)for(j=1;j<=m+i-1;++j)ins(num[i][j],num[i][j]+cc,1,0); for(i=1;i<=n+m-1;++i) ins(num[n][i]+cc,T1,1,mun[n][i]); for(i=1;i <=i+m-1;++j){ ins(num[i][j]+cc,num[i+1][j],1,mun[i][j]); ins(num[i][j]+cc,num[i+1][j+1],1,mun[i][j]); }ins(T1,T,m,0); solve(); printf("%d\n",mincost); memset(hr,0,sizeof(hr));cnt=1;mincost=maxflow=0; for(i=1;i<=m;++i) ins(S,num[1][i],1,0); for(i=1;i<=n+m-1;++i) ins(num[n][i],T1,inf,mun[n][i]); for(i=1;i
17.运输问题
费用流裸题。最大费用+最小费用。
#includeusing namespace std;inline int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}#define MN 105#define ME 20500#define S 0#define T 205#define inf 0x3f3f3f3fint maxflow,mincost;struct edge{ int to,w,c,nex;}e[ME];int hr[T+5],cnt=1;inline void ins(int f,int t,int w,int c){ e[++cnt]=(edge){t,w,c,hr[f]};hr[f]=cnt; e[++cnt]=(edge){f,0,-c,hr[t]};hr[t]=cnt; }int d[T+5],q[ME*2+5],l,r;bool inq[T+5],vis[T+5];bool spfa(){ memset(d,0x3f,sizeof d); q[l=r=ME]=T;d[T]=0;inq[T]=1; while(l<=r){ int u=q[l++];inq[u]=0; for(int i=hr[u];i;i=e[i].nex) if(e[i^1].w&&d[e[i].to]>d[u]-e[i].c){ d[e[i].to]=d[u]-e[i].c; if(!inq[e[i].to]) d[e[i].to]
18.分配问题
二分图最佳匹配?只会大最小费用最大流。
#includeusing namespace std;inline int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}#define MN 105#define ME 20500#define S 0#define T 205#define inf 0x3f3f3f3fint maxflow,mincost;struct edge{ int to,w,c,nex;}e[ME];int hr[T+5],cnt=1;inline void ins(int f,int t,int w,int c){ e[++cnt]=(edge){t,w,c,hr[f]};hr[f]=cnt; e[++cnt]=(edge){f,0,-c,hr[t]};hr[t]=cnt; }int d[T+5],q[ME*2+5],l,r;bool inq[T+5],vis[T+5];bool spfa(){ memset(d,0x3f,sizeof d); q[l=r=ME]=T;d[T]=0;inq[T]=1; while(l<=r){ int u=q[l++];inq[u]=0; for(int i=hr[u];i;i=e[i].nex) if(e[i^1].w&&d[e[i].to]>d[u]-e[i].c){ d[e[i].to]=d[u]-e[i].c; if(!inq[e[i].to]) d[e[i].to]
19.负载平衡问题
所有相邻仓库之间连容量inf,费用为1的边,S向所有仓库连容量为库存数量,费用为0的边,所有仓库向T连容量为平均数,费用为0的边。
#includeusing namespace std;inline int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}#define MN 105#define ME 1500 #define S 0#define T 105#define inf 0x3f3f3f3fint maxflow,mincost;struct edge{ int to,w,c,nex;}e[ME];int hr[T+5],cnt=1;inline void ins(int f,int t,int w,int c){ e[++cnt]=(edge){t,w,c,hr[f]};hr[f]=cnt; e[++cnt]=(edge){f,0,-c,hr[t]};hr[t]=cnt; }int d[T+5],q[ME*2+5],l,r;bool inq[T+5],vis[T+5];bool spfa(){ memset(d,0x3f,sizeof d); q[l=r=ME]=T;d[T]=0;inq[T]=1; while(l<=r){ int u=q[l++];inq[u]=0; for(int i=hr[u];i;i=e[i].nex) if(e[i^1].w&&d[e[i].to]>d[u]-e[i].c){ d[e[i].to]=d[u]-e[i].c; if(!inq[e[i].to]) d[e[i].to] n?1:i+1,inf,1),ins(i,i-1<1?n:i-1,inf,1); solve(); printf("%d\n",mincost); return 0;}
20.深海机器人问题
最大费用最大流,有生物标本的点需要拆点,然后连费用为1,流量为1的边就行了。
#includeusing namespace std;inline int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}#define MN 16#define ME 20000#define S 400#define T 405#define inf 200000000int mincost,maxflow;struct edge{ int to,w,c,nex;}e[ME+5];int hr[T+5],q[ME*2+5],d[T+5],vis[T+5],inq[T+5],cnt=1,l,r,u;inline void ins(int f,int t,int w,int c){ e[++cnt]=(edge){t,w,c,hr[f]};hr[f]=cnt; e[++cnt]=(edge){f,0,-c,hr[t]};hr[t]=cnt;}inline bool spfa(){ memset(d,128,sizeof d); q[l=r=ME+1]=T;inq[T]=1;d[T]=0; while(l<=r){ int u=q[l++];inq[u]=0; for(int i=hr[u];i;i=e[i].nex) if(e[i^1].w&&d[e[i].to] d[q[l]]?q[--l]=e[i].to:q[++r]=e[i].to,inq[e[i].to]=1; } } return d[S]>d[403];}inline int mcf(int x,int f){ vis[x]=1; if(x==T) return f; int used=0,w; for(int i=hr[x];i;i=e[i].nex) if(!vis[e[i].to]&&d[e[i].to]==d[x]-e[i].c&&e[i].w){ w=mcf(e[i].to,min(f-used,e[i].w)); used+=w;e[i].w-=w;e[i^1].w+=w; mincost+=e[i].c*w; if(f==used) return used; } return used;}inline void solve(){ while(spfa()){ do{ memset(vis,0,sizeof vis); maxflow+=mcf(S,inf); }while(vis[T]); }}int a,b,P,Q;inline int pos(int x,int y){ return (Q+1)*x+y;}int main(){ a=read(),b=read(),P=read(),Q=read(); register int i,j,x,y,k; for(i=0;i<=P;i++)for(j=1;j<=Q;j++) ins(pos(i,j-1),pos(i,j),1,read()),ins(pos(i,j-1),pos(i,j),inf,0); for(j=0;j<=Q;j++)for(i=1;i<=P;i++) ins(pos(i-1,j),pos(i,j),1,read()),ins(pos(i-1,j),pos(i,j),inf,0); for(i=1;i<=a;i++)k=read(),x=read(),y=read(),ins(S,pos(x,y),k,0); for(i=1;i<=b;i++)k=read(),x=read(),y=read(),ins(pos(x,y),T,k,0); solve(); printf("%d\n",mincost); return 0; }
21.最长k可重区间集问题
建图方式:
然后跑最大费用最大流。
#includeusing namespace std;inline int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}#define T 1005#define S 0#define ME 5000#define inf 200000000int mincost,maxflow;struct edge{ int to,w,c,nex;}e[ME+5];int cnt=1,l,u,r,d[T+5],inq[T+5],vis[T+5],q[ME*2+5],hr[T+5];inline void ins(int f,int t,int w,int c){ e[++cnt]=(edge){t,w,c,hr[f]};hr[f]=cnt; e[++cnt]=(edge){f,0,-c,hr[t]};hr[t]=cnt;}inline bool spfa(){ memset(d,128,sizeof d); q[l=r=ME+1]=T;inq[T]=1;d[T]=0; while(l<=r){ int u=q[l++];inq[u]=0; for(int i=hr[u];i;i=e[i].nex) if(e[i^1].w&&d[e[i].to] d[q[l]]?q[--l]=e[i].to:q[++r]=e[i].to,inq[e[i].to]=1; } } return d[S]>d[1003];}inline int mcf(int x,int f){ vis[x]=1; if(x==T) return f; int used=0,w; for(int i=hr[x];i;i=e[i].nex) if(!vis[e[i].to]&&d[e[i].to]==d[x]-e[i].c&&e[i].w){ w=mcf(e[i].to,min(f-used,e[i].w)); used+=w;e[i].w-=w;e[i^1].w+=w; mincost+=e[i].c*w; if(f==used) return used; } return used;}inline void solve(){ while(spfa()){ do{ memset(vis,0,sizeof vis); maxflow+=mcf(S,inf); }while(vis[T]); }}int n,k;int nd[T+5],rk[T+5],pos[T+5];bool cmp(int a,int b){ return nd[a]
22.最长k可重线段集问题
和第21题类似。
由于线段可能垂直于x轴,导致首尾两点作对的x坐标相等,所以我们需要拆点一下。
#includeusing namespace std;inline long long read(){ long long x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}#define T 2005#define S 0#define ME 100000#define inf 200000000#define int long longint mincost,maxflow;struct edge{ int to,w,c,nex;}e[ME+5];int cnt=1,l,u,r,d[T+5],inq[T+5],vis[T+5],q[ME*2+5],hr[T+5];inline void ins(int f,int t,int w,int c){ e[++cnt]=(edge){t,w,c,hr[f]};hr[f]=cnt; e[++cnt]=(edge){f,0,-c,hr[t]};hr[t]=cnt;}inline bool spfa(){ memset(d,128,sizeof d); q[l=r=ME+1]=T;inq[T]=1;d[T]=0; while(l<=r){ int u=q[l++];inq[u]=0; for(int i=hr[u];i;i=e[i].nex) if(e[i^1].w&&d[e[i].to] d[q[l]]?q[--l]=e[i].to:q[++r]=e[i].to,inq[e[i].to]=1; } } return d[S]>d[2003];}inline int mcf(int x,int f){ vis[x]=1; if(x==T) return f; int used=0,w; for(int i=hr[x];i;i=e[i].nex) if(!vis[e[i].to]&&d[e[i].to]==d[x]-e[i].c&&e[i].w){ w=mcf(e[i].to,min(f-used,e[i].w)); used+=w;e[i].w-=w;e[i^1].w+=w; mincost+=e[i].c*w; if(f==used) return used; } return used;}inline void solve(){ while(spfa()){ do{ memset(vis,0,sizeof vis); maxflow+=mcf(S,inf); }while(vis[T]); }}int n,k;int nd[T+5],ndy[T+5],rk[T+5],pos[T+5];bool cmp(int a,int b){ return nd[a]
23.火星探险问题
和深海机器人问题差不多的。最大费用最大流。
#includeusing namespace std;inline int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}#define inf 0x3f3f3f3f#define S 0#define T 2455#define MN 40#define ME 12000int maxflow,mincost;struct edge{ int to,w,c,nex;}e[ME+5]; int d[T+5],q[ME*2],hr[T+5],cnt=1,l,r;bool vis[T+5],inq[T+5];inline void ins(int f,int t,int w,int c){ e[++cnt]=(edge){t,w,c,hr[f]};hr[f]=cnt; e[++cnt]=(edge){f,0,-c,hr[t]};hr[t]=cnt;}inline bool spfa(){ memset(d,128,sizeof d); q[l=r=ME]=T;inq[T]=1;d[T]=0; while(l<=r){ int u=q[l++];inq[u]=0; for(int i=hr[u];i;i=e[i].nex) if(e[i^1].w&&d[e[i].to] d[q[l]]?q[--l]=e[i].to:q[++r]=e[i].to,inq[e[i].to]=1; } } return d[S]>0;}inline int mcf(int x,int f){ vis[x]=1; if(x==T) return f; int used=0,w; for(int i=hr[x];i;i=e[i].nex) if(!vis[e[i].to]&&d[e[i].to]==d[x]-e[i].c&&e[i].w){ w=mcf(e[i].to,min(f-used,e[i].w)); used+=w;e[i].w-=w;e[i^1].w+=w; mincost+=e[i].c*w; if(f==used) return used; } return d[x]=-1,used;}inline void solve(){ while(spfa()){ do{ memset(vis,0,sizeof vis); maxflow+=mcf(S,inf); }while(vis[T]); }}int n,m,k,tot;int mp[MN][MN],pos[MN][MN],step[MN][MN];inline void dfs(int id,int x,int y){ if(x==n&&y==m) return; if(step[x+1][y]&&x+1<=n&&y<=m){ // printf("%d %d\n",x+1,y); printf("%d 0\n",id); step[x+1][y]--; dfs(id,x+1,y); } else if(step[x][y+1]&&x<=n&&y+1<=m){ // printf("%d %d\n",x,y+1); printf("%d 1\n",id); step[x][y+1]--; dfs(id,x,y+1); }}int main(){ k=read(),m=read(),n=read(); register int i,j; for(i=1;i<=n;i++) for(j=1;j<=m;j++){ pos[i][j]=(i-1)*m+j; mp[i][j]=read(); if(mp[i][j]==1) mp[i][j]=-1; if(mp[i][j]==2) mp[i][j]=++tot; } for(i=1;i<=n;i++)for(j=1;j<=m;j++)if(mp[i][j]>=0){ if(i =0) ins(pos[i][j],pos[i+1][j],inf,0); if(j =0) ins(pos[i][j],pos[i][j+1],inf,0); if(mp[i][j]>0){ ins(pos[i][j],n*m+mp[i][j],1,1); if(i =0) ins(n*m+mp[i][j],pos[i+1][j],inf,0); if(j =0) ins(n*m+mp[i][j],pos[i][j+1],inf,0); } } ins(S,1,k,0);ins(n*m,T,inf,0);if(mp[n][m]>0) ins(mp[n][m]+n*m,T,inf,0); solve(); for(i=1;i<=n;i++)for(j=1;j<=m;j++)for(int p=hr[pos[i][j]];p;p=e[p].nex)if(p&1) step[i][j]+=e[p].w; for(i=1;i<=maxflow;i++){ dfs(i,1,1); } return 0;}
24.骑士共存问题
最小割。首先将图进行黑白染色,然后将会相互攻击到的点之间连一条inf的边,S向黑点连流量为1的边,白点向T连流量为1的边。跑最大流。
#includeusing namespace std;inline int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){ if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();} return x*f;}#define MN 205#define T 40005#define S 0#define ME 500000#define inf 200000000#define p pd(i,j,x,y)int cnt=1,d[T+5],hr[T+5],cur[T+5],q[T+5],top,ans;struct edge{ int to,w,nex;}e[ME];inline void ins(int f,int t,int w){ e[++cnt]=(edge){t,w,hr[f]};hr[f]=cnt; e[++cnt]=(edge){f,0,hr[t]};hr[t]=cnt;}inline bool bfs(){ memset(d,0,sizeof d); register int i,j; for(d[q[i=top=1]=S]=1;i<=top;i++) for(j=hr[q[i]];j;j=e[j].nex) if(e[j].w&&!d[e[j].to]) d[q[++top]=e[j].to]=d[q[i]]+1; return d[T];}inline int dfs(int x,int f){ if(x==T) return f;int used=0; for(int i=cur[x];i;i=e[i].nex){ cur[x]=i; if(e[i].w&&d[e[i].to]==d[x]+1){ int w=dfs(e[i].to,min(f-used,e[i].w)); e[i].w-=w,used+=w,e[i^1].w+=w; if(used==f) return used; } } return d[x]=-1,used;}inline void dinic(){ while(bfs()){ memcpy(cur,hr,sizeof(int)*(T+5)); ans+=dfs(S,inf); }}int n,m;bool mp[MN][MN];int g(int x,int y){ return (x-1)*n+y;}void pd(int i,int j,int x,int y){ if(x>0&&x<=n&&y>0&&y<=n&&!mp[x][y]) ins(g(i,j),g(x,y),inf);}int main(){ n=read(),m=read(); register int i,j,x,y; for(i=1;i<=m;i++) x=read(),mp[x][read()]=1; for(i=1;i<=n;i++)for(j=1;j<=n;j++)if(!mp[i][j]) (i^j)&1?ins(S,g(i,j),1):ins(g(i,j),T,1); for(i=1;i<=n;i++)for(j=1;j<=n;j++)if(!mp[i][j]&&(i^j)&1){ x=i-2;y=j+1;p; x=i-2;y=j-1;p; x=i-1;y=j+2;p; x=i-1;y=j-2;p; x=i+2;y=j+1;p; x=i+2;y=j-1;p; x=i+1;y=j+2;p; x=i+1;y=j-2;p; } dinic(); printf("%d\n",n*n-m-ans); return 0;}
来自PaperCloud的博客,未经允许,请勿转载,TKS。