洛谷P4307 球隊收益

題意:有n個球隊,m場比賽。ide

每一個球隊都已經有些勝負場次了。spa

每一個球隊的收益爲Ci * wini2 - Di * losei2code

求最小可能總收益。blog

解:string

先看出一個模型:用一流量表明一個勝場,每場比賽向兩支隊伍連邊。it

而後咱們發現這個費用是跟流量的平方有關的,How to do?io

先觀察一波:1 4 9 16 25event

差分:1 3 5 7 9class

而後咱們就發現:若是把下面差分建成邊的費用,限流爲1,剛好就是收益了。cli

至此茅塞頓開。

首先假設全部的隊伍都輸了,而後每場選出一名勝者,C(2win + 1) - D(2lose - 1)爲費用。

最小費用最大流便可。

  1 #include <cstdio>
  2 #include <algorithm>
  3 #include <queue>
  4 #include <cstring>
  5 
  6 const int N = 6050, M = 1000010;
  7 const int INF = 0x3f3f3f3f;
  8 
  9 struct Edge {
 10     int nex, v;
 11     int c, len;
 12 }edge[M << 1]; int top = 1;
 13 
 14 int e[N], vis[N], pre[N];
 15 int d[N], flow[N];
 16 std::queue<int> Q;
 17 int A[N], B[N], C[N], D[N], win[N], los[N], X[N], Y[N];
 18 
 19 inline void add(int x, int y, int z, int w) {
 20     //printf("add : %d %d \n", x, y);
 21     top++;
 22     edge[top].v = y;
 23     edge[top].c = z;
 24     edge[top].len = w;
 25     edge[top].nex = e[x];
 26     e[x] = top;
 27 
 28     top++;
 29     edge[top].v = x;
 30     edge[top].c = 0;
 31     edge[top].len = -w;
 32     edge[top].nex = e[y];
 33     e[y] = top;
 34     return;
 35 }
 36 
 37 inline bool SPFA(int s, int t) {
 38     memset(d, 0x3f, sizeof(d));
 39     d[s] = 0;
 40     flow[s] = INF;
 41     vis[s] = 1;
 42     Q.push(s);
 43     while(!Q.empty()) {
 44         int x = Q.front();
 45         Q.pop();
 46         vis[x] = 0;
 47         //printf("x = %d d = %d\n", x, d[x]);
 48         for(int i = e[x]; i; i = edge[i].nex) {
 49             int y = edge[i].v;
 50             if(edge[i].c && d[y] > d[x] + edge[i].len) {
 51                 d[y] = d[x] + edge[i].len;
 52                 pre[y] = i;
 53                 flow[y] = std::min(flow[x], edge[i].c);
 54                 if(!vis[y]) {
 55                     vis[y] = 1;
 56                     Q.push(y);
 57                 }
 58             }
 59         }
 60     }
 61     return d[t] < INF;
 62 }
 63 
 64 inline void update(int s, int t) {
 65     int temp = flow[t];
 66     while(t != s) {
 67         int i = pre[t];
 68         edge[i].c -= temp;
 69         edge[i ^ 1].c += temp;
 70         t = edge[i ^ 1].v;
 71     }
 72     return;
 73 }
 74 
 75 inline int solve(int s, int t, int &cost) {
 76     int ans = 0;
 77     cost = 0;
 78     while(SPFA(s, t)) {
 79         ans += flow[t];
 80         cost += flow[t] * d[t];
 81         update(s, t);
 82     }
 83     return ans;
 84 }
 85 
 86 int main() {
 87     int n, m, s, t, sum = 0;
 88     scanf("%d%d", &n, &m);
 89     s = n + m + 1, t = m + n + 2;
 90     for(int i = 1; i <= n; i++) {
 91         scanf("%d%d%d%d", &win[i], &los[i], &C[i], &D[i]);
 92         /*win[i] = A[i];
 93         los[i] = B[i];*/
 94     }
 95     for(int i = 1; i <= m; i++) {
 96         scanf("%d%d", &X[i], &Y[i]);
 97         add(s, n + i, 1, 0);
 98         add(n + i, X[i], 1, 0);
 99         add(n + i, Y[i], 1, 0);
100         los[X[i]]++;
101         los[Y[i]]++;
102     }
103     for(int i = 1; i <= n; i++) {
104         sum += C[i] * win[i] * win[i];
105         sum += D[i] * los[i] * los[i];
106     }
107     for(int i = 1; i <= m; i++) {
108         int x = X[i], y = Y[i];
109         add(x, t, 1, C[x] * (2 * win[x] + 1) - D[x] * (2 * los[x] - 1));
110         add(y, t, 1, C[y] * (2 * win[y] + 1) - D[y] * (2 * los[y] - 1));
111         win[x]++;
112         win[y]++;
113         los[x]--;
114         los[y]--;
115     }
116 
117     int ans;
118     solve(s, t, ans);
119     printf("%d", ans + sum);
120     return 0;
121 }
AC代碼
相關文章
相關標籤/搜索