免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
12下一页
最近访问板块 发新帖
查看: 7450 | 回复: 10
打印 上一主题 下一主题

[C] 两个宏定义的整数值太大,导致计算错误????高手帮看看 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2010-02-04 20:39 |只看该作者 |倒序浏览
如题,我定义了两个宏定义,一个是int的正整数最大整数值的一半,另一个是long的正整数值(大于int的正整数最大值),然后两个进行比较。做这个的本意是想验证数值比较的时候会默认将类型转换为int,结果它们的确是往int转了,但比较结果缺和预想的不一样,代码如下,大家帮我看看:

  1 #include <stdio.h>
  2
  3 #define MINT                           1222483647
  4 #define MLONGLONG   2141111111345342
  5
  6 int main(void)
  7 {
  8     if (MINT < MLONGLONG)
  9     {
10         printf("mint < mlonglong\n");
11     }
12     else if (MINT > MLONGLONG)
13     {
14         printf("mint > mlonglong\n");
15     }
16
17     return 0;
18 }
19

编译的信息:
test.c:8: 警告:对 ‘long’ 类型而言整数常量太大
test.c:12: 警告:对 ‘long’ 类型而言整数常量太大

程序是按小端来填数据的,按照编译的信息来看MLONGLONG是按long来转换的,int和long都是32位,装不下这么大的数,所以把它截下来了,结果
MLONGLONG应该是1111345342,而MINT是1222483647。

因此,应该MINT > MLONGLONG, 但最后的打印结果刚好相反:mint < mlonglong。

请高手帮忙解释下

论坛徽章:
0
2 [报告]
发表于 2010-02-04 20:40 |只看该作者
溢出了

论坛徽章:
2
程序设计版块每日发帖之星
日期:2015-06-17 22:20:00每日论坛发贴之星
日期:2015-06-17 22:20:00
3 [报告]
发表于 2010-02-04 20:41 |只看该作者
提示: 作者被禁止或删除 内容自动屏蔽

论坛徽章:
0
4 [报告]
发表于 2010-02-05 00:24 |只看该作者
        [code][/code]movl        $1222483647, 28(%esp)
        movl        $1194812606, 16(%esp)
        movl        $498516, 20(%esp)
        movl        28(%esp), %eax
        movl        %eax, %edx
        sarl        $31, %edx
        cmpl        20(%esp), %edx


把MINT右移了31位再和MLONGLONG的高位比较,相当于比较了高位!
这个warning很容易误导人...

论坛徽章:
0
5 [报告]
发表于 2010-02-05 10:03 |只看该作者
如果是需要使用比较大的数,你应该这么定义
#define MINT                           1222483647lu
#define MLONGLONG   2141111111345342llu

至于大常量溢出之后,行为是不确定。错误的前提推出的结论能靠得住么?

论坛徽章:
4
2015年辞旧岁徽章
日期:2015-03-03 16:54:152015年迎新春徽章
日期:2015-03-04 09:56:11IT运维版块每日发帖之星
日期:2016-08-11 06:20:00IT运维版块每日发帖之星
日期:2016-08-15 06:20:00
6 [报告]
发表于 2010-02-05 10:49 |只看该作者
如果是需要使用比较大的数,你应该这么定义
#define MINT                           1222483647lu
#defi ...
pagx 发表于 2010-02-05 10:03


楼上正解!
大数后不带修饰符,编译器会按int类型对待的,那就溢出了!

论坛徽章:
0
7 [报告]
发表于 2010-02-05 15:10 |只看该作者
学习

论坛徽章:
2
程序设计版块每日发帖之星
日期:2015-06-17 22:20:00每日论坛发贴之星
日期:2015-06-17 22:20:00
8 [报告]
发表于 2010-02-05 21:05 |只看该作者
提示: 作者被禁止或删除 内容自动屏蔽

论坛徽章:
0
9 [报告]
发表于 2010-02-05 21:20 |只看该作者
{:2_174:}

最近感觉高手好多啊。

我是来学习的。

论坛徽章:
0
10 [报告]
发表于 2010-02-06 08:49 |只看该作者
楼上正解!
大数后不带修饰符,编译器会按int类型对待的,那就溢出了!
happy_fish100 发表于 2010-02-05 10:49



自动转成int型,某些编译器会这样来优化,标准应该是没有的。gcc好像是某些版本会有
讨论这个问题没太大意义,规范的代码应该要带ul这些后缀,并且到调用时再显式转换一次,就比较靠谱了。
不要嫌难看,代码打扮得再漂亮,凭它也泡不到妞
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP