P2294 [HNOI2005]狡猾的商人
题目描述

输入输出格式
输入格式:
从文件input.txt中读入数据,文件第一行为一个正整数w,其中w < 100,表示有w组数据,即w个账本,需要你判断。每组数据的第一行为两个正整数n和m,其中n < 100,m < 1000,分别表示对应的账本记录了多少个月的收入情况以及偷看了多少次账本。接下来的m行表示刁姹偷看m次账本后记住的m条信息,每条信息占一行,有三个整数s,t和v,表示从第s个月到第t个月(包含第t个月)的总收入为v,这里假设s总是小于等于t。
输出格式:
输出文件output.txt中包含w行,每行是true或false,其中第i行为true当且仅当第i组数据,即第i个账本不是假的;第i行为false当且仅当第i组数据,即第i个账本是假的。
输入输出样例
2
3 3
1 2 10
1 3 -5
3 3 -15
5 3
1 5 100
3 5 50
1 2 51
true
false
Solution:
本题是思维比较巧妙的差分约束。。。
思路类似于植树那道题,$u\rightarrow v=c$可以理解为$sum[v]-sum[u-1]=c$(前缀和)。
那么对于每个条件我们可以先得出约束条件:$w[u-1,v]=c$表示$sum[v]$比$sum[u-1]$大$c$。
我们直接在这样的图上跑最长路后能求出每个点的$dis$值,但是如何去判断是否合法呢?
先看这张图(手绘勿喷):

此图表示的是$sum[1,3]=1,\;sum[1,2]=1,\;sum[2,3]=1$的情况,显然是不合法的情况,但是我们求出最长路后的$dis$值分别为$dis[1]=0,\;dis[2]=1\;dis[3]=2$,貌似对判断合法没什么用。
不难发现,用少了约束条件,给定的$sum[u\rightarrow v]=c$不仅需要满足$sum[v]-sum[u-1]=c$,还应该满足$sum[u-1]-sum[v]=-c$。
那么我们加入这一约束后,图就变成了这样:

此时,我们由第一张图求得的$dis[1]=0,\;dis[2]=1,\;dis[3]=2$在此图中还可以继续更新,$dis[1]=0$会由$dis[3]$更新变为$dis[1]=1$,然后引发一系列无限制的更新。究其原因,就是因为不满足$sum[u-1]-sum[v]=-c$这一约束条件。
所以我们将两个约束条件都建成边。
再考虑另一个问题,那就是源点不确定,所以我们每次以没有被遍历过的点为源点跑$spfa$求最长路(即使约束条件中没有出现过该点也对答案无影响,因为根本不会由它进行广搜遍历),那么当某个点被重复遍历超过$n$次,则出现了环,说明不可行,否则就是合法的。(事实证明数据比较水,因为我第一遍没有考虑源点不为$0$的情况,但是却$A$了~~)
代码:
#include<bits/stdc++.h>
#define il inline
#define For(i,a,b) for(int (i)=(a);(i)<=(b);(i)++)
using namespace std;
const int N=,inf=;
int T,to[N],tot[N],net[N],h[N],w[N],dis[N],cnt,n,m;
bool vis[N];
queue<int>q; il int gi(){
int a=;char x=getchar();bool f=;
while((x<''||x>'')&&x!='-')x=getchar();
if(x=='-')x=getchar(),f=;
while(x>=''&&x<='')a=(a<<)+(a<<)+x-,x=getchar();
return f?-a:a;
} il void add(int u,int v,int c){to[++cnt]=v,net[cnt]=h[u],h[u]=cnt,w[cnt]=c;} il bool spfa(int s){
For(i,,n)dis[i]=-inf;
q.push(s);dis[s]=;vis[s]=;
while(!q.empty()){
int u=q.front();q.pop();vis[u]=;tot[u]++;
if(tot[u]==n)return ;
for(int i=h[u];i;i=net[i])
if(dis[to[i]]<dis[u]+w[i]){
dis[to[i]]=dis[u]+w[i];
if(!vis[to[i]])q.push(to[i]),vis[to[i]]=;
}
}
return ;
} int main(){
T=gi();
while(T--){
n=gi(),m=gi();
cnt=;
memset(vis,,sizeof(vis));
memset(h,,sizeof(h));
memset(tot,,sizeof(tot));
int u,v,c,f;
while(m--){
u=gi(),v=gi(),c=gi();
add(u-,v,c);add(v,u-,-c);
}
f=;
For(i,,n-){
if(!tot[i]){
if(!spfa(i)){f=;break;}
}
}
if(!f)printf("true\n");
else printf("false\n");
}
return ;
}
P2294 [HNOI2005]狡猾的商人的更多相关文章
- [luogu P2294] [HNOI2005]狡猾的商人
[luogu P2294] [HNOI2005]狡猾的商人 题目描述 输入输出格式 输入格式: 从文件input.txt中读入数据,文件第一行为一个正整数w,其中w < 100,表示有w组数据, ...
- 洛谷P2294 [HNOI2005]狡猾的商人
P2294 [HNOI2005]狡猾的商人 题目描述 输入输出格式 输入格式: 从文件input.txt中读入数据,文件第一行为一个正整数w,其中w < 100,表示有w组数据,即w个账本,需要 ...
- P2294 [HNOI2005]狡猾的商人(差分约束)
P2294 [HNOI2005]狡猾的商人 对于每个$(x,y,w)$,连边$(x-1,y,w),(y,x-1,-w)$,表示前$y$个月的收益比前$x-1$个月的收益大$w$ 这样题目就转化为询问图 ...
- LUOGU P2294 [HNOI2005]狡猾的商人(差分约束)
[传送门] (https://www.luogu.org/problemnew/show/P2294) 解题思路 差分约束.先总结一下差分约束,差分约束就是解决一堆不等式混在一起,左边是差的形式,右边 ...
- 洛谷 [p2294] [HNOI2005] 狡猾的商人
差分约束做法 又是一道转换成前缀和的差分约束题,已知从s月到t月的收入w,设数组pre[i]代表从开始到第i个月的总收入 构造差分不等式 $ pre[s-1]-pre[t]==w $ 为了满足松弛操作 ...
- 题解——洛谷P2294 [HNOI2005]狡猾的商人(差分约束)
裸的差分约束 dfs判断负环,如果有负环就false,否则就是true 注意有多组数据,数组要清空 #include <cstdio> #include <algorithm> ...
- Bzoj1202/洛谷P2294 [HNOI2005]狡猾的商人(带权并查集/差分约束系统)
题面 Bzoj 洛谷 题解 考虑带权并查集,设\(f[i]\)表示\(i\)的父亲(\(\forall f[i]<i\)),\(sum[i]\)表示\(\sum\limits_{j=fa[i]} ...
- [HNOI2005]狡猾的商人 ,神奇做法——贪心
洛谷P2294 [HNOI2005]狡猾的商人 ,神奇做法--贪心 看到大牛都是写的差分约束或带权并查集,本蒟蒻都不太会(还是用差分约束过了的QAQ),但是想出一种贪心的策略,运用神奇的优先队列实现. ...
- [BZOJ1202][HNOI2005]狡猾的商人
[BZOJ1202][HNOI2005]狡猾的商人 试题描述 刁姹接到一个任务,为税务部门调查一位商人的账本,看看账本是不是伪造的.账本上记录了n个月以来的收入情况,其中第i 个月的收入额为Ai(i= ...
随机推荐
- ios数据持久化--CoreData框架的介绍和使用
1.Core Data 是数据持久化存储的最佳方式 2.数据最终的存储类型可以是:SQLite数据库,XML,二进制,内存里,或自定义数据类型 在Mac OS X 10.5Leopard及以后的版本中 ...
- LeetCode804. Unique Morse Code Words
题目 国际摩尔斯密码定义一种标准编码方式,将每个字母对应于一个由一系列点和短线组成的字符串, 比如: "a" 对应 ".-", "b" 对应 ...
- Python——列表
应用场景,要统计大量的人员信息.就可以用列表的形式进行.name = ['邱秀','玄永俊','杨栋豪']查:print(name[2],name[0]) #取号码进行识别print(name[1:3 ...
- js、jquery中全局替换replace
str.replace(/需要替换的/g,"新字符串") //此处使用正则表达式
- Spring Cloud 入门 Consul-Client服务提供
前面介绍了 Rureka Client服务提供, 只需要改pom.xml部分内容 1.pom.xml <?xml version="1.0" encoding="U ...
- js将人民币数字转大写
function numberToUpper(money) { var cnNums = new Array("零", "壹", "贰", ...
- ofbiz研究
近段时间,刚有有时间研究了下ofbiz ; 目前还是刚开始,后期会记录过程 有一起研究的没
- PLC状态机编程-如何在STL中使用状态机
搞PLC编程多年,一直不知道状态机,学习matlab后,发现状态机编程异常方便,过去很多编程时的疑惑豁然开朗起来.今天跟大家分享一下如何在STL中使用状态机. 下面是用状态机描述的控制任务. 这个状态 ...
- pytorch中词向量生成的原理
pytorch中的词向量的使用 在pytorch我们使用nn.embedding进行词嵌入的工作. 具体用法就是: import torch word_to_ix={'hello':0,'world' ...
- 散列--数据结构与算法JavaScript描述(8)
散列 散列是一种常用的数据存储技术,散列后的数据可以快速地插入或取用. 散列使用的数据结构叫做散列表. 在散列表上插入.删除和取用数据都非常快,但是对于查找操作来说却效率低下,比如查找一组数据中的最大 ...