C++之路进阶——差分约束(糖果)

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C++之路进阶——差分约束(糖果)相关的知识,希望对你有一定的参考价值。

2404 糖果

 

2011年省队选拔赛四川

 时间限制: 1 s
 空间限制: 128000 KB
 题目等级 : 大师 Master
 
 
题目描述 Description

幼儿园里有N个小朋友,lxhgww老师现在想要给这些小朋友们分配糖果,要求每个小朋友都要分到糖果。但是小朋友们也有嫉妒心,总是会提出一些要求,比如小明不希望小红分到的糖果比他的多,于是在分配糖果的时候,lxhgww需要满足小朋友们的K个要求。幼儿园的糖果总是有限的,lxhgww想知道他至少需要准备多少个糖果,才能使得每个小朋友都能够分到糖果,并且满足小朋友们所有的要求。

输入描述 Input Description

输入的第一行是两个整数N,K。

接下来K行,表示这些点需要满足的关系,每行3个数字,X,A,B。

如果X=1, 表示第A个小朋友分到的糖果必须和第B个小朋友分到的糖果一样多;

如果X=2, 表示第A个小朋友分到的糖果必须少于第B个小朋友分到的糖果;

如果X=3, 表示第A个小朋友分到的糖果必须不少于第B个小朋友分到的糖果;

如果X=4, 表示第A个小朋友分到的糖果必须多于第B个小朋友分到的糖果;

如果X=5, 表示第A个小朋友分到的糖果必须不多于第B个小朋友分到的糖果;

输出描述 Output Description

输出一行,表示lxhgww老师至少需要准备的糖果数,如果不能满足小朋友们的所有要求,就输出-1

样例输入 Sample Input

5 7

1 1 2

2 3 2

4 4 1

3 4 5

5 4 5

2 3 5

4 5 1

样例输出 Sample Output

11

数据范围及提示 Data Size & Hint

对于30%的数据,保证 N<=100

 

对于100%的数据,保证 N<=100000

 

对于所有的数据,保证 K<=100000,1<=X<=5,1<=A, B<=N

代码:

 1 #include<cstdio>
 2 #define maxn 1000100
 3 #include<iostream>
 4 
 5 using namespace std;
 6 
 7 int cnt;
 8 long long ans;
 9 
10 struct cf
11    {
12         int n,m,head[maxn],v[maxn],to[maxn],next[maxn],rvs[maxn],ins[maxn],dis[maxn],q[maxn*10];
13         void insert(int u,int vv,int w) {to[++cnt]=vv;next[cnt]=head[u];head[u]=cnt; v[cnt]=w;}
14         bool spfa()
15            {
16                  int t=0,w=1;
17                 q[0]=0;
18                 ins[0]=1;
19                 rvs[0]=1;
20                while (t<w)
21                   {
22                         int now=q[t++];
23                         for (int i=head[now];i;i=next[i])
24                                  if (dis[to[i]]<dis[now]+v[i]) 
25                                     {   
26                                 if (++rvs[to[i]]>=n) return 0;
27                                 dis[to[i]]=dis[now]+v[i];
28                                 if (!ins[to [i]]){ins[to[i]]=1;q[w++]=to[i];} 
29                              }                                                                   
30                         ins[now]=0; 
31                   }
32             return 1;      
33             } 
34         int start()
35         {    
36            scanf("%d%d",&n,&m);    
37            for (int i=1;i<=m;i++)     
38               {
39                   int f,a,b;
40                   scanf("%d%d%d",&f,&a,&b);
41                   switch(f)
42                      {
43                          case 1:insert(a,b,0); insert(b,a,0); break;
44                          case 2:if (a==b) {printf("-1\n");return 0;} insert(a,b,1);break;
45                          case 3:insert(b,a,0);break;
46                          case 4:if (a==b) {printf("-1\n");return 0;} insert(b,a,1);break;
47                           case 5:insert(a,b,0); break;     
48                       }           
49               }
50            for (int i=n;i>=1;i--) insert(0,i,1);      
51            if (!spfa()) {printf("-1\n");return 0;}
52            for (int i=1;i<=n;i++) ans+=dis[i];      
53            printf("%lld\n",ans);     
54           }        
55    }cf;
56 int main()
57   {
58       cf.start();
59     return 0;
60   }

 

 

以上是关于C++之路进阶——差分约束(糖果)的主要内容,如果未能解决你的问题,请参考以下文章

bzoj2330: [SCOI2011]糖果(差分约束)

bzoj 2330[SCOI2011]糖果 - 差分约束

bzoj2330[SCOI2011]糖果 差分约束系统

[SCOI2011]糖果 [差分约束]

bzoj2330糖果——差分约束

CDOJ 435 (SCOI 2011) 糖果 Label:差分约束系统