[转]ACM中WA方面的错误总结

1、 Int遇到过的问题

简介:

int是我们最常用的类型之一。如果输入数据是整形,一般都直接用该类型来存放输入数据。


错误经历:

自己在作Equiptment Box时,因为输入数据长、宽均是小于50000的整数,因此就使用int来作输入。这本身没有问题,但在求其斜边长时,使用的是sqrt(x * x + y * y),表面看是没有问题,但结果一直是Wrong Answer。 后来将这一行改为pow( (pow(x, 2) + pow( y,3 ) ) , 0.5),就Accept了!

 

错误原因:

    后来经johnbill和hewei的分析,x,y本身没有问题,不会越界,但使用sqrt(x*x +y*y)时,里面的x*x 和 y*y则会超出int范围,造成溢出。而pow会将参数自动转换为double,就不会出错。


避免失误的办法:

(1)以后均使用pow进行运算。(习惯)

(2)运算时,注意做强行转换。(比较麻烦)

(3)不管输入给的类型,直接用double来存储,就不会溢出了。这种方法表面看没有问题,但直到这次比赛,才发现了一个很严重的问题!

 

2、 double遇到过的问题

简介:

是我们在解题时,和int一起是最常用的类型。

 

错误经历:

因为 double上限可达1.7e308  。而一般题目(非大数运算要求)均不可能超过其限,发生溢出,所以之后我就在做题时,凡是遇到结果有些大时,均用double类型来保存,来避免溢出。看起来,这样比较方便,因为我们在本机上是用VC++,而OnlineJudge是gcc,它们支持的长整形类型不同,一个是__int64, 而一个是long long;处理格式也不同,I

64u 和 lld。而在这种情况下,“真正”的可以用double的话,那就可以将其统一起来。但是……昨天比赛C题时,自己也是这么递推和用double保存,但一直Wrong Answer。和递归能计算出(太大的数据很耗时)的数据相比,都是正确的,不知原因何在。比赛结束后,和别人结果对照了一下,把double改成unsigned long long 就Accept了。

 

错译原因:

这是因为:double类型的精度只有15位!!!它的上限可以很大,但只能保证15位的精度!换句话说,只能保证15位是正确的。在数据( 50,50)以后,结果都在20位以上,前面的位数是正确的,但后面的几位就会出现问题了!

 

 解决方法:

(1)    定义头文件,在本机上用__int64,提交时用long long

(2)    本机上使用VAC编译( J ) (奇难用!)

(3)    反正绝对不能使用double来计数,尤其比较大的数,但可以利用它来测试最大数据的范围大小,这样可以反过来帮助我们决定用什么类型来保存。

 

3、    float遇到过的问题

我还记得当时Hunter做area的时候,各方面都作了考虑,但一直是Wrong Answer。后来只是把存储坐标的float类型改为double,就过了。

   

原因:

应该是float的精度不够(具体嘛…..大家re),但题目只要求3位小数也有问题……。所以,以后大家要使用浮点数计算时,直接用double,不要考虑使用float。一般内存是不会有问题的。

 

4、    4舍5入的问题:

在做Lifting the Stone时,题上要求保留到小数点两位,第三位作四舍五入,自己直接用%.02来打,以为自动会四舍五入。但一直没过。加上处理之后就过了…….

 

原因:

小数点后第三位为5时,会随机的作进位处理。

 

解决办法:

如果题上要求了四舍五入,一定要记得进行处理:x = floor(x*100 + 0.5)/100,

 

5、    为5时,后一位奇数进位,偶数不进位:

 这个JohnBIll讲过,一般不会有这种“浪费青春的题…..”。

 

解决办法:

除了相当的灵活,多试,还需要运气了……

 

posted @ 2010-09-11 15:41  勇泽  阅读(701)  评论(0编辑  收藏  举报