2 5 1 1 10 1 1 1 1 1 5 1 1 500 1 1 1 1 1
Case #1: 2 Case #2: 7
AC代码:
#include <stdio.h> #include <string.h> #include <stdlib.h> int main() { int testcase,i,n,count,ans; double k,sum,k1,k2,temp; scanf("%d",&testcase); for(count=1; count<=testcase; count++) { ans = 0; sum = 0; scanf("%d %lf %lf %lf",&n,&k1,&k2,&k); for(i=0;i<n;i++) { scanf("%lf",&temp); sum = sum + temp; } if(sum > k) printf("Case #%d: 0\n",count); else if(k1+k2>1 || k1+k2<-1) { while(sum<=k) { sum = sum * ( k1 + k2 ); ans++; } printf("Case #%d: %d\n",count,ans); } else { printf("Case #%d: inf\n",count); } } return 0; }
题目分析:此题难度不大,但是AC率极低,因为题中存在一个很隐蔽的陷阱。
由题目描述,我们可以抽象出公式sum(i+1) = (k1+k2)*sum(i),目标是当sum大于k时,则停止生长;若sum永远不可能大于k,则无法停止生长,直接输出inf即可。
由于-100 <= k1, k2 <= 100,所以k1+k2可正可负也可为0。当k1+k2等于0,-1或1时,显然永远无法停止生长,输出inf。其余情况,则需要根据sum(i+1) = (k1+k2)sum(i)计算即可,数一下迭代的次数,即可算得多少天之后可停止生长。
现在来说一下此题的陷阱:64位整型在运算时会溢出。k的取值范围是1 <= k <= 10 ^ 18,这个数超过了32位整型变量,但是用64位整型可表示。带符号的64位整型最大值为2^63 - 1,这个数大于10 ^ 18,乍一看,好像不会有什么问题。但是公式sum(i+1) = (k1+k2)*sum(i)在迭代的时候,会超过2^63 - 1,也就说会溢出发生错误。原因是sum(i)理论上最大可与k相等,k1+k2的最大值为200,相乘得200*10 ^ 18大于2^63 - 1,这证明了极端情况会发生溢出。所以说用64位整型变量会出现错误,应该用double表示才可,尽管输入都是整数。我当时做此题时,一直不明白错在哪里,比赛结束之后才发现问题所在。
如果非要使用64位整型变量也可,需要程序做一下改动,在相乘的时候不要发生溢出,也可AC,代码如下:
#include <stdio.h> #include <string.h> #include <stdlib.h> int main() { int testcase,i,n,count,ans; __int64 k,sum,k1,k2,temp; scanf("%d",&testcase); for(count=1; count<=testcase; count++) { ans = 1; sum = 0; scanf("%d %I64d %I64d %I64d",&n,&k1,&k2,&k); for(i=0;i<n;i++) { scanf("%I64d",&temp); sum = sum + temp; } if(sum > k) printf("Case #%d: 0\n",count); else if(k1+k2>1 || k1+k2<-1) { while(1.0*(k1+k2)*sum<=1.0*k) { sum = sum * ( k1 + k2 ); ans++; } printf("Case #%d: %d\n",count,ans); } else { printf("Case #%d: inf\n",count); } } return 0; }