6"):puts("<=6");}C语言中的整数自动转换原则,我发现有些开发者懂得极少这些东西。不管如何,这无符号整型问题的答案是输出是"> 6"):puts("<=6");}C语言中的整数自动转换原则,我发现有些开发者懂得极少这些东西。不管如何,这无符号整型问题的答案是输出是" />
欢迎来到天天文库
浏览记录
ID:8808003
大小:93.00 KB
页数:11页
时间:2018-04-08
《c数据类型转换及操作》由会员上传分享,免费在线阅读,更多相关内容在应用文档-天天文库。
1、voidfoo(void){unsignedinta=6;intb=-20;(a+b>6)puts(">6"):puts("<=6");}C语言中的整数自动转换原则,我发现有些开发者懂得极少这些东西。不管如何,这无符号整型问题的答案是输出是“>6”。原因是当表达式中存在有符号类型和无符号类型时所有的操作数都自动转换为无符号类型。因此-20变成了一个非常大的正整数,所以该表达式计算出的结果大于6。这一点对于应当频繁用到无符号数据类型的嵌入式系统来说是丰常重要的。如果你答错了这个问题,你也就到了得不到这份工作的边缘在汇编语言层面,声明变量的时候,没有signed和unsignde之分,汇编器统统,
2、将你输入的整数字面量当作有符号数处理成补码存入到计算机中,只有这一个标准!汇编器不会区分有符号还是无符号然后用两个标准来处理,它统统当作有符号的!并且统统汇编成补码!也就是说,db-20汇编后为:EC,而db236汇编后也为EC。这里有一个小问题,思考深入的朋友会发现,db是分配一个字节,那么一个字节能表示的有符号整数范围是:-128~+127,那么db236超过了这一范围,怎么可以?是的,+236的补码的确超出了一个字节的表示范围,那么拿两个字节(当然更多的字节更好了)是可以装下的,应为:00EC,也就是说+236的补码应该是00EC,一个字节装不下,但是,别忘了“截断”这个概念,就是说最后
3、汇编的结果被截断了,00EC是两个字节,被截断成EC,所以,这是个“美丽的错误”,为什么这么说?因为,当你把236当作无符号数时,它汇编后的结果正好也是EC,这下皆大欢喜了,虽然汇编器只用一个标准来处理,但是借用了“截断”这个美丽的错误后,得到的结果是符合两个标准的!也就是说,给你一个字节,你想输入有符号的数,比如-20那么汇编后的结果是符合有符号数的;如果你输入236那么你肯定当作无符号数来处理了(因为236不在一个字节能表示的有符号数的范围内啊),得到的结果是符合无符号数的。于是给大家一个错觉:汇编器有两套标准,会区分有符号和无符号,然后分别汇编。其实,你们被骗了。:-)3.第一点说明汇编
4、器只用一个方法把整数字面量汇编成真正的机器数。但并不是说计算机不区分有符号数和无符号数,相反,计算机对有符号和无符号数区分的十分清晰,因为计算机进行某些同样功能的处理时有两套指令作为后备,这就是分别为有符号和无符号数准备的。但是,这里要强调一点,一个数到底是有符号数还是无符号数,计算机并不知道,这是由你来决定的,当你认为你要处理的数是有符号的,那么你就用那一套处理有符号数的指令,当你认为你要处理的数是无符号的,那就用处理无符号数的那一套指令。加减法只有一套指令,因为这一套指令同时适用于有符号和无符号。下面这些指令:muldivmovzx…是处理无符号数的,而这些:imulidivmovsx…是
5、处理有符号的。举例来说:内存里有一个字节x为:0xEC,一个字节y为:0x02。当把x,y当作有符号数来看时,x=-20,y=+2。当作无符号数看时,x=236,y=2。下面进行加运算,用add指令,得到的结果为:0xEE,那么这个0xEE当作有符号数就是:-18,无符号数就是238。所以,add一个指令可以适用有符号和无符号两种情况。(呵呵,其实为什么要补码啊,就是为了这个呗,:-))乘法运算就不行了,必须用两套指令,有符号的情况下用imul得到的结果是:0xFFD8就是-40。无符号的情况下用mul,得到:0x01D8就是472。4.为什么又扯到c了?因为大多数遇到有符号还是无符号问题的朋
6、友,都是c里面的signed和unsigned声明引起的,那为什么开头是从汇编讲起呢?因为我们现在用的c编译器,无论gcc也好,vc6的cl也好,都是将c语言代码编译成汇编语言代码,然后再用汇编器汇编成机器码的。搞清楚了汇编,就相当于从根本上明白了c,而且,用机器的思维去考虑问题,必须用汇编。(我一般遇到什么奇怪的c语言的问题都是把它编译成汇编来看。)C是可爱的,因为c符合kiss原则,对机器的抽象程度刚刚好,让我们即提高了思维层面(比汇编的机器层面人性化多了),又不至于离机器太远(像c#,java之类就太远了)。当初K&R版的c就是高级一点的汇编,C又是可怕的,因为它把机器层面的所有的东西都
7、反应了出来,像这个有没有符号的问题就是一例(java就不存在这个问题,因为它被设计成所有的整数都是有符号的)。为了说明它的可怕特举一例:#include#includeintmain(){intx=2;char*str="abcd";inty=(x-strlen(str))/2;printf("%d",y);}结果应该是-1但是却得到:2147483647
此文档下载收益归作者所有