codeforces 434D

题意:有n<=50个点,每个点有xi有[li, ri]种取值,-100 <= li <= ri <= 100,并且给定m<=100条边,每条边为u,v,d表示xu<=xv+d。

        每个点value fi(x) = ai*x^2 + bi*x + ci。现在求一种合法的方案,使得权值和最大。

思路:先不考虑的xu<=xv + d。那么建图:

       首先考虑到每个点的权值可能为负,并且求最大与最小割相反,

        所以先 取反再+oo(一个很大的数),最后再减掉即可

        对于每个点,拆成ri-li+1个点,

        对于第k个点,node(k, i)表示第k个点值为i对应的标号

        值为i-1跟i连一条边<node(k, i-1), node(k, i),  oo - f(k, i)>的边,

        S到第一个点连<S, node(k, l[k]), f(k, l[k])>

       最后一个点到T连<node(k,r[k]), Inf>

       那么很明显n*oo-最小割就是答案。。

       但是如果有了限制条件xu<=xv + d,我们怎么把限制条件加到图上呢?

       对于一对关系,xu<=xv+d

       考虑到点u,如果node(u, i)到node(u,i+1)之间的边在割集里,那么说明xu=i+1

       也就是说如果xv<xu-d是非法的,也就是说对于v在xu-d之前出现割是非法的。

       那么我们可以连<node(u,i), node(v, i-d), Inf>的边,使得方案合法。。

code:

 1 #include <bits/stdc++.h>
 2 using namespace std;
 3 #define M0(a) memset(a, 0, sizeof(a))
 4 #define Inf 0x3fffffff
 5 const int maxn = 20010;
 6 const int maxm = 900010;
 7 const int big = 15000000;
 8 struct oo{
 9     int y, f, next;
10 };
11 struct MaxFlow{
12        int n, S, T, tot;
13        int son[maxn], dist[maxn], gap[maxn];
14        oo e[maxm];
15        int sap(int x, int aug){
16            if (x == T) return aug;
17            int mind = n;
18            int sum = 0, f;
19            for (int p = son[x]; p != -1; p = e[p].next){
20                   int y = e[p].y;
21                   if (dist[y] + 1 == dist[x] && e[p].f){
22                        f = sap(y, min(e[p].f, aug - sum));
23                        e[p].f -= f;
24                        e[p^1].f += f;
25                        sum += f;
26                        if (sum == aug || dist[S] >= n) return sum;
27                   }
28                   if (e[p].f) mind = min(mind, dist[y]);
29            }
30            if (!sum){
31                if (!(--gap[dist[x]])) dist[S] = n;
32                ++gap[dist[x] = mind + 1];
33            }
34            return sum;
35        }
36        void add(int x, int y, int f){
37             e[tot].y = y; e[tot].f = f;
38             e[tot].next = son[x]; son[x] = tot++;
39             e[tot].y = x; e[tot].f = 0;
40             e[tot].next = son[y]; son[y] = tot++;
41        }
42 
43        void init(int S, int T, int n){
44             memset(son, -1, sizeof(son));
45             tot = 0;
46             this->S = S, this->T = T, this->n = n;
47        }
48        int maxflow(){
49             M0(gap);
50             M0(dist);
51             gap[0] = n;
52             int ans = 0;
53             while (dist[S] < n) ans += sap(S, Inf);
54             return ans;
55        }
56 } F;
57 int S, T;
58 int n, m, a[120], b[200], c[200], l[200], r[200];
59 inline int f(const int&k, const int& x){
60      return big - (a[k] * x * x + b[k] * x + c[k]);
61 }
62 
63 inline int node(const int&k, const int& x){
64      return x == l[k] - 1 ? S : (k-1) * 201 + x + 101;
65 }
66 
67 void solve(){
68      for (int i = 1; i <= n; ++i) scanf("%d%d%d", &a[i], &b[i], &c[i]);
69      for (int i = 1;  i<= n; ++i) scanf("%d%d", &l[i], &r[i]);
70      S = 0, T = 201 * n + 1;
71      F.init(S, T, T + 1);
72      for (int i = 1; i <= n; ++i){
73          for (int j = l[i]; j <= r[i]; ++j)
74              F.add(node(i, j-1), node(i, j), f(i, j));
75          F.add(node(i, r[i]), T, Inf);
76      }
77      int u, v, d;
78      int x;
79      for (int i = 1; i <= m; ++i){
80           scanf("%d%d%d", &u, &v, &d);
81           for (int j = l[v]; j <= r[v]; ++j) if (j + d <= r[u]){
82                   x = j + d;
83                   if (x < l[u]) x = l[u] -1;
84                   F.add(node(u, x), node(v, j), Inf); 
85           }
86      }
87      int ans = big * n;
88      ans -= F.maxflow();
89      cout << ans << endl;
90 }
91 
92 int main(){
93 //    freopen("a.in", "r", stdin);
94     while (scanf("%d%d", &n, &m) != EOF){
95         solve();
96     }            
97 }
View Code

 

posted on 2014-10-30 11:07  yzcstc  阅读(348)  评论(0编辑  收藏  举报