題意:有n個球隊,m場比賽。ide
每一個球隊都已經有些勝負場次了。spa
每一個球隊的收益爲Ci * wini2 - Di * losei2。code
求最小可能總收益。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 }