BZOJ1202

来源:互联网 发布:wordpress yoast seo 编辑:程序博客网 时间:2024/06/07 02:50

1202: [HNOI2005]狡猾的商人

Time Limit: 10 Sec  Memory Limit: 162 MB
Submit: 1339  Solved: 636
[Submit][Status]

Description

刁姹接到一个任务,为税务部门调查一位商人的账本,看看账本是不是伪造的。账本上记录了n个月以来的收入情况,其中第i 个月的收入额为Ai(i=1,2,3...n-1,n), 。当 Ai大于0时表示这个月盈利Ai 元,当 Ai小于0时表示这个月亏损Ai 元。所谓一段时间内的总收入,就是这段时间内每个月的收入额的总和。 刁姹的任务是秘密进行的,为了调查商人的账本,她只好跑到商人那里打工。她趁商人不在时去偷看账本,可是她无法将账本偷出来,每次偷看账本时她都只能看某段时间内账本上记录的收入情况,并且她只能记住这段时间内的总收入。 现在,刁姹总共偷看了m次账本,当然也就记住了m段时间内的总收入,你的任务是根据记住的这些信息来判断账本是不是假的。

Input

第一行为一个正整数w,其中w < 100,表示有w组数据,即w个账本,需要你判断。每组数据的第一行为两个正整数n和m,其中n < 100,m < 1000,分别表示对应的账本记录了多少个月的收入情况以及偷看了多少次账本。接下来的m行表示刁姹偷看m次账本后记住的m条信息,每条信息占一行,有三个整数s,t和v,表示从第s个月到第t个月(包含第t个月)的总收入为v,这里假设s总是小于等于t。

Output

包含w行,每行是true或false,其中第i行为true当且仅当第i组数据,即第i个账本不是假的;第i行为false当且仅当第i组数据,即第i个账本是假的。

Sample Input

2
3 3
1 2 10
1 3 -5
3 3 -15
5 3
1 5 100
3 5 50
1 2 51

Sample Output

true
false

【题解】我们可以这样想,账本能够判定为假的条件就是,有三段的和,并且最长段可以由另外两个拼起来,满足sum[3]!=sum[2]+sum[1];所以我用了一个数组f [ i ][ j ],记录i 到 j 的和为多少。然后考虑衔接,我们先将所有段按s 为第一关键字,t 为第二关键字从小到大排序,然后枚举 j, if(f[ j ][a[i].s-1]!=初值) 存在可以衔接的对象,在判断 if(f [ i ][ a[i].t ]==初值)则直接赋值,否则判断是否相等,不等则说明是假的。


#include<cstdio>#include<cstring>#include<algorithm>using namespace std;struct node{int s,t,sum;}a[1000];int f[100][100];inline bool cmp(node x,node y){if(x.s<y.s)return 1; else if(x.s>y.s)return 0;  else return x.t<y.t;}int main(){int w,inf,n,m;scanf("%d",&w);while(w--){memset(f,0x7f,sizeof(f));inf=f[1][1];scanf("%d%d",&n,&m);for(int i=1;i<=m;i++)scanf("%d%d%d",&a[i].s,&a[i].t,&a[i].sum);sort(a+1,a+1+m,cmp);bool key=0;for(int i=1;i<=m;i++){if(f[a[i].s][a[i].t]==inf)f[a[i].s][a[i].t]=a[i].sum; else if(f[a[i].s][a[i].t]!=a[i].sum){key=1;break;} for(int j=1;j<=a[i].s-1;j++)  if(f[j][a[i].s-1]!=inf)   if(f[j][a[i].t]==inf)f[j][a[i].t]=f[j][a[i].s-1]+a[i].sum;    else if(f[j][a[i].t]!=f[j][a[i].s-1]+a[i].sum)key=1;    if(key)break;}if(key)printf("false\n"); else printf("true\n");}return 0;}



0 0