poj 3189 Steady Cow Assignment(二分+最大流)
生活随笔
收集整理的這篇文章主要介紹了
poj 3189 Steady Cow Assignment(二分+最大流)
小編覺得挺不錯的,現在分享給大家,幫大家做個參考.
題意:N頭牛(1000),B個農場(20),每個農場可以容納一定數量的牛。
每頭牛對每個農場都有一個排名(排名從1~B)。每頭牛都會在B個農場中的某一個,這頭牛的高興程度是它對這個農場的排名。為了使每頭牛都盡量同等高興,希望所有牛中最高興的和最不高興的程度差值最小,求這個差值。
構圖:二分最小差,對每一個最小差,枚舉起點、終點。最大流判可行性。
#include<iostream> #include<cstdio> #include<cstring> #include<queue> using namespace std;const int maxn = 1500; const int INF = 0x3f3f3f3f; struct Edge {int to,next,flow; }edge[maxn*150]; int n,m,Rank[maxn][25]; int cnt,head[maxn],num[maxn]; int level[maxn],st,ed;void addedge(int u,int v,int flow) {edge[cnt].to = v;edge[cnt].flow = flow;edge[cnt].next = head[u];head[u] = cnt++;swap(u,v);edge[cnt].to = v;edge[cnt].flow = 0;edge[cnt].next = head[u];head[u] = cnt++; }void build(int l,int r) //rank[l]~rank[r]之間 {cnt = 0;memset(head,-1,sizeof(head));for(int i = 1; i <= n; i++) addedge(st,i,1);for(int i = 1; i <= n; i++)for(int j = l; j <= r; j++)addedge(i,n + Rank[i][j],1);for(int i = n + 1; i <= n + m; i++) addedge(i,ed,num[i-n]); }int BFS(int src,int des){queue<int >q;memset(level,0,sizeof(level));level[src]=1;q.push(src);while(!q.empty()){int u = q.front();q.pop();if(u==des) return 1;for(int k = head[u];k!=-1;k=edge[k].next){int v = edge[k].to,w=edge[k].flow;if(level[v]==0&&w!=0){level[v]=level[u]+1;q.push(v);}}}return -1; }int dfs(int u,int des,int increaseRoad){if(u==des) return increaseRoad;int ret=0;for(int k=head[u];k!=-1;k=edge[k].next){int v = edge[k].to,w=edge[k].flow;if(level[v]==level[u]+1&&w!=0){int MIN = min(increaseRoad-ret,w);w = dfs(v,des,MIN);if(w > 0){edge[k].flow -= w;edge[k^1].flow += w;ret+=w;if(ret==increaseRoad) return ret;}else level[v] = -1; }}return ret; } int Dinic(int src,int des){int ans = 0;while(BFS(src,des)!=-1) ans+=dfs(src,des,INF);return ans; }bool check(int len) {for(int i = 1; i + len - 1 <= m; i++){build(i,i+len-1);int tmp = Dinic(st,ed);if(tmp == n) return true;}return false; }int main() {while(scanf("%d%d",&n,&m)!=EOF){st = 0, ed = n + m + 1;for(int i = 1; i <= n; i++)for(int j = 1; j <= m; j++)scanf("%d",&Rank[i][j]);for(int i = 1; i <= m; i++)scanf("%d",&num[i]);int l = 1,r = m,mid,ans;while(l <= r){mid = (l + r) >> 1;if(check(mid)){ans = mid;r = mid - 1;}else l = mid + 1;}printf("%d\n",ans);}return 0; }
總結
以上是生活随笔為你收集整理的poj 3189 Steady Cow Assignment(二分+最大流)的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: springMVC整合swagger(亲
- 下一篇: JEECG Excel 介绍篇