差分超级坑题--nkoj2112(scoi2011)

Problem C:【SCOI2011 Day1】糖果

Time Limit:10000MS  Memory Limit:165536K
Total Submit:308 Accepted:74 
Case Time Limit:3000MS

Description

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

Input

  输入的第一行是两个整数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

  输出一行,表示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

Hint

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


注意使用long long

重要结论1:
以Xi-Xj<=C为约束条件,建图求最短路后得到的是最大解。
所有解都不大于且尽可能逼进Dis[X0]。


若有Xi-Xj<=C,可在图中连一条从Xj出发指向Xi的有向边,边的权值为C。再添加一个源点X0,从X0出发往X1,X2,...,Xn连一条权值为0的有向边。
然后以X0为起点,求单源最短路(无负权环),求出的起点X0到每个点的距离Dis[ ]就是不等式组的一组解。
若一开始就把Dis[X0]的值定死为A,再求最短路,那么求出的其它点的Dis[ ]值一定不会大于A。并且,求出的Dis[ ]是不大于A且与A最接近的一组。也就是所有点的Dis[ ]都达到了最大,也称为最大解。


若有Xi-Xj>=C,可在图中连一条从Xj出发指向Xi的有向边,边的权值为C。再添加一个源点X0,从X0出发往X1,X2,...,Xn连一条权值为0的有向边。


然后以X0为起点,求单源最长路(无正权环),求出的起点X0到每个点的距离Dis[ ]就是不等式组的一组解。


若一开始就把Dis[X0]的值定死为A,再求最长路,那么求出的其它点的Dis[ ]值一定不会小于A。并且,求出的Dis[ ]是不小于A且与A最接近的一组。也就是所有点的Dis[ ]都达到了最小,也称为最小解。


重要结论2:
以Xi-Xj>=C为约束条件,建图求最长路后得到的是最小解。
所有解都不小于且尽可能逼进Dis[X0]。

差分约束小结

最短路:(对应最大解)
1.按Xi-Xj<=C的条件建图(从Xj出发向Xi连接权值为C的有向边)。
2.添加超级源点X0,从X0出发往每个点连接一条权值为0的边
3.固定Dis[X0]的值为A
4.以X0为源点求最短路(无负权回路的前提下),求出的每个点的Dis[ ]对应一组解。
5.所有的Dis[ ]一定都是<=A的最大解。


最长路:(对应最小解)
1.按Xi-Xj>=C的条件建图(从Xj出发向Xi连接权值为C的有向边)。
2.添加超级源点X0,从X0出发往每个点连接一条权值为0的边
3.固定Dis[X0]的值为A
4.以X0为源点求最长路(无正权回路的前提下),求出的每个点的Dis[ ]对应一组解。
5.所有的Dis[ ]一定都是>=A的最小解。

#include<cstdio>
#include<iostream>
#include<algorithm>
#include<cstring>
#include<queue>
const int inf=-1e9;
using namespace std;
long long dis[1000005],next[4000005],last[1000005];
long long cnt[1000005];
bool vis[1000005];
long long m=0;
long long n,k;
queue<long long>q;
struct edge{
	long long from,to,len;
	edge(){}
	edge(long long a,long long b,long long c){from=a;to=b;len=c;}
};
edge line[4000005];
inline void read(long long &x){  
    char t;  
    bool mark=false;  
    for(;t=getchar(),t<'0'||t>'9';) if(t=='-') mark=1;  
    for(x=t-'0',t=getchar();'0'<=t&&t<='9';x=x*10+t-'0',t=getchar());  
    x=mark?-x:x;  
}
void add_edge(long long from,long long to,long long len){
	m++;
	next[m]=last[from];
	last[from]=m;
	line[m]=edge(from,to,len);
}
bool spfa(int s){
	long long i,j,k,t,ans=0;
	for(i=1;i<=n;i++)dis[i]=inf;
	vis[s]=true;
	dis[s]=1; 
	cnt[s]=1;
	q.push(s);
	while(q.size()){
		t=q.front();q.pop();vis[t]=false;
		for(i=last[t];i;i=next[i]){
			if(dis[line[i].to]<dis[line[i].from]+line[i].len){
				dis[line[i].to]=dis[line[i].from]+line[i].len;
				if(vis[line[i].to]==false){
					cnt[line[i].to]++;
					if(cnt[line[i].to]>=n){
						return false;
					}
					q.push(line[i].to);
					vis[line[i].to]=true;
				}
			}
		}
	}
	return true;
}
int main(){
	long long i,j,t,x,y,ans=0;
	cin>>n>>k;
	for(i=1;i<=k;i++){
		read(t);read(x);read(y);
		if(t==1){
			add_edge(x,y,0);
			add_edge(y,x,0);
		}
		else if(t==2){
			if(x==y){
				cout<<"-1";return 0;
			}
			add_edge(x,y,1);
		}
		else if(t==3){
			add_edge(y,x,0);
		}
		else if(t==4){
			if(x==y){
				cout<<"-1";return 0;
			}
			add_edge(y,x,1);
		}
		else{
			add_edge(x,y,0);
		}
	}
	for(i=n;i>=1;i--)add_edge(n+1,i,0);//数据有鬼,不能从1--n
	bool flag=spfa(n+1);
	if(flag){
		for(i=1;i<=n;i++)ans=ans+dis[i];
		cout<<ans;
	}
	else cout<<"-1";
	return 0;
} 


你可能感兴趣的:(差分超级坑题--nkoj2112(scoi2011))