P1993 小K的农场

题目描述

小K在MC里面建立很多很多的农场,总共n个,以至于他自己都忘记了每个农场中种植作物的具体数量了,他只记得一些含糊的信息(共m个),以下列三种形式描述:

  • 农场a比农场b至少多种植了c个单位的作物,
  • 农场a比农场b至多多种植了c个单位的作物,
  • 农场a与农场b种植的作物数一样多。

但是,由于小K的记忆有些偏差,所以他想要知道存不存在一种情况,使得农场的种植作物数量与他记忆中的所有信息吻合。

输入格式

第一行包括两个整数 n 和 m,分别表示农场数目和小 K 记忆中的信息数目。

接下来 m 行:

如果每行的第一个数是 1,接下来有 3 个整数 a,b,c,表示农场 a 比农场 b 至少多种植了 c 个单位的作物。

如果每行的第一个数是 2,接下来有 3 个整数 a,b,c,表示农场 a 比农场 b 至多多种植了 c 个单位的作物。如果每行的第一个数是 3,接下来有 2 个整数 a,b,表示农场 a 种植的的数量和 b 一样多。

输出格式

如果存在某种情况与小 K 的记忆吻合,输出“Yes”,否则输出“No”。

输入输出样例

输入 #1
3 3
3 1 2
1 1 3 1
2 2 3 2
输出 #1
Yes

说明/提示

对于 100% 的数据保证:1 ≤ n,m,a,b,c ≤ 10000。

思路

源自:five20

代码

#include<cmath>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;

const int N=50010,oo=1e9;

bool vis[N];
int n,m,tot;
int to[N],hed[N],w[N];
int sum[N],dis[N],nxt[N];

void add(int u,int v,int c) {
	to[++tot]=v;
	nxt[tot]=hed[u];
	hed[u]=tot;
	w[tot]=c;
}

bool spfa(int u) {
	vis[u]=1;
	for(int i=hed[u]; i; i=nxt[i])
		if(dis[to[i]]<dis[u]+w[i]) {
			dis[to[i]]=dis[u]+w[i];
			if(vis[to[i]])
				return false;
			if(!spfa(to[i]))
				return false;
		}
	vis[u]=0;
	return true;
}

int main () {
	scanf("%d%d",&n,&m);
	int f,u,v,w;
	while(m--) {
		scanf("%d%d%d",&f,&u,&v);
		if(f==1) {
			scanf("%d",&w);
			add(u,v,w);
		}
		if(f==2) {
			scanf("%d",&w);
			add(u,v,-w);
		}
		if(f==3) {
			add(u,v,0);
			add(v,u,0);
		}
	}
	for(int i=1; i<=n; i++) {
		add(0,i,0);
		dis[i]=-oo;
	}
	if(!spfa(0))
		printf("No\n");
	else
		printf("Yes\n");
	return 0;
}

 

posted @ 2019-09-16 00:33  双子最可爱啦  阅读(114)  评论(0编辑  收藏  举报