标签:收集 问题 bsp 经验 mil ems 分数 题目 描述
题目描述
输入
输出
样例输入
3
19 17 16
25 24 23
35 36 31
9 5 6
3 4 2
7 8 9
样例输出
5.357143
题解
分数规划+费用流
二分答案mid,将每个点的值看作a-b*mid。
由于每个男生只能搭配一名不同的女生,所以问题可以转化为:1个n*n的矩阵中每个位置都有1个数,求选出n个彼此不在同一行或同一列的数的和的最大值是多少。
加边s->i(1,0),i+n->t(1,0),i->j+n(1,v[i][j]),跑最大费用最大流,若大于0则调整下界,否则调整上界,直至上下界基本重合。
证明:如果最大费用大于0,则∑(v[ik][jk])>0,即∑(a[ik][jk]-b[ik][jk]*mid)>0,即∑a[ik][jk]>∑b[ik][jk]*mid,即∑a[ik][jk]/∑b[ik][jk]>mid,故需要调整下界来进一步更新答案,否则调整上界来调整答案。
#include <cstdio> #include <cstring> #include <queue> #define eps 1e-7 using namespace std; queue<int> q; int n , head[210] , to[30000] , val[30000] , next[30000] , cnt , s , t , from[210] , pre[210]; double a[110][110] , b[110][110] , cost[30000] , dis[210]; void add(int x , int y , int v , double c) { to[++cnt] = y , val[cnt] = v , cost[cnt] = c , next[cnt] = head[x] , head[x] = cnt; to[++cnt] = x , val[cnt] = 0 , cost[cnt] = -c , next[cnt] = head[y] , head[y] = cnt; } bool spfa() { int x , i; for(i = s ; i <= t ; i ++ ) from[i] = -1 , dis[i] = 10000000.0; dis[s] = 0 , q.push(s); while(!q.empty()) { x = q.front() , q.pop(); for(i = head[x] ; i ; i = next[i]) if(val[i] && dis[to[i]] > dis[x] + cost[i]) dis[to[i]] = dis[x] + cost[i] , from[to[i]] = x , pre[to[i]] = i , q.push(to[i]); } return ~from[t]; } double mincost() { int k , i; double ans = 0; while(spfa()) { k = 0x3f3f3f3f; for(i = t ; i != s ; i = from[i]) k = min(k , val[pre[i]]); ans += k * dis[t]; for(i = t ; i != s ; i = from[i]) val[pre[i]] -= k , val[pre[i] ^ 1] += k; } return ans; } bool judge(double mid) { int i , j; memset(head , 0 , sizeof(head)); cnt = 1; for(i = 1 ; i <= n ; i ++ ) add(s , i , 1 , 0); for(i = 1 ; i <= n ; i ++ ) add(i + n , t , 1 , 0); for(i = 1 ; i <= n ; i ++ ) for(j = 1 ; j <= n ; j ++ ) add(i , j + n , 1 , mid * b[i][j] - a[i][j]); return mincost() < 0; } int main() { int i , j; double l = 0.0 , r = 10000000.0 , mid; scanf("%d" , &n); s = 0 , t = 2 * n + 1; for(i = 1 ; i <= n ; i ++ ) for(j = 1 ; j <= n ; j ++ ) scanf("%lf" , &a[i][j]); for(i = 1 ; i <= n ; i ++ ) for(j = 1 ; j <= n ; j ++ ) scanf("%lf" , &b[i][j]); while(l <= r) { mid = (l + r) / 2; if(judge(mid)) l = mid + eps; else r = mid - eps; } printf("%.6lf\n" , (l + r) / 2); return 0; }
【bzoj4819】[Sdoi2017]新生舞会 分数规划+费用流
标签:收集 问题 bsp 经验 mil ems 分数 题目 描述
原文地址:http://www.cnblogs.com/GXZlegend/p/6856213.html