C语言细节——献给入门者(三)
>>主题:关于强制类型转换
先来瞎扯下强制类型转换,c语言有很多数据类型,long,short,int,float,double,bool,char等等。当我们要将一种数据类型赋值给另一种数据类型,或者将某个操作数进行如此的转化的时候,就用到了强制类型转换。
强制类型转换分成两种:1.显示强制类型转换(主动) 2.隐式强制类型转换(被动/自动)
要区分这两者很简单,看下面例子:(we assume that b is a variable of integer.)
char a = (char)b; char a = b;
所以区分两者只要看这个(char)是否显式的写出来。但我们要注意,这两个的区别绝对不是这么简单。让我们看下去...
》》》如果一个运算符两边的运算数类型不同,先要将其转换为相同的类型,即较低类型转换为较高类型,然后再参加运算,转换规则如下图所示。
这段和这图是我百度来的,他说计算机会主动把float,int分别转换成double,long计算,后来想来一下发现跟计算机组成学的不一样,况且如果都这样计算,那还需要小数据类型干嘛呢?计算机速度何必非得用双精度来测量?后来我自己测了一下,发现果然是瞎扯,我就不用这个图说啦~(附上测试代码)
(ps:以下按照int为32位来说明):
测试代码:
#include <stdio.h> int main() { int a=1,b=2; int c=a+b; long aa=2,cc=4; long bb=aa+cc; printf("%d ",c ); printf("%ld ",bb ); return 0; }
反汇编后所得(部分代码):
1 movl $0, -4(%rbp) 2 movl $1, -8(%rbp) 3 movl $2, -12(%rbp) 4 movl -8(%rbp), %eax 5 addl -12(%rbp), %eax 6 movl %eax, -16(%rbp) 7 movq $2, -24(%rbp) 8 movq $4, -32(%rbp) 9 movq -24(%rbp), %rcx 10 addq -32(%rbp), %rcx 11 movq %rcx, -40(%rbp) 12 movl -16(%rbp), %esi 13 movb $0, %al
感觉一下子打乱我思路了,让我捋一捋。。。
》》》先看下两个优先级的例子,很多人可能没有注意到:
num = (double)k/m;
在这里,无论double是强制类型转换了k还是k/m,答案都一样;但这边的double是修饰k的,因为括号的优先级是高于/的;
如果想要double修饰k/m的话就可以这样做:
num = (double)(k/m);
另一个例子:
a = (int)father->age;
这里很明显int是修饰age的,因为father是不能强制类型转换成int类型的;这里是因为->的优先级高于括号;
》》》接下来看看有关数据类型的几种例子
case 1:
int a,b=2; a=(int)b;
这里 因为a,b都是int型,所以这边的(int)可以不用,即显式隐式都是可以的;
case 2:
int a; char b='A'; a=(int)b;
char a; int b=65; a=(char)b;
首先这个例子能互换是因为字符和整型之间可以通过ascii码转换,其次char占一个字节而int占4个字节;
在前一个大数给小数的情况下,显式隐式编译器都通过,但是很有可能导致溢出,所以需要自己注意;
后一个系统自动会把b扩成4字节赋值给a(b还是1字节)
case 3:
int b=-1; unsigned int a; a=b; printf("%d ",a);
这里unsigned int的范围是0~65535 ,其实当a=b后,a的值是65535,但输出却是-1;我一开始没有想通,加了显式转换之后试了一下还是-1,后来才知道%d输出是有符号的int类型,所以输出结果还是-1;
说到这个我想到之前一个学弟问我%e的问题;
那是书上的一个例子,输入用%e,输出却用%f,他一直奇怪为什么输出是71.820000而不是7.182000e+01;
你不用%e输出,怎么会看到7.182000e+01呢?就算你用%f输入%e输出也能看到这样的形式啊。
我觉得学语言不能光学个语法,重点需要了解数据是如何存在于计算机内部的。
你需要知道无论怎么样输入输出,同样的数据存在计算机里面是不变的,只是我们在屏幕上看到的显示方式不一样而已,就像一个月饼,有包装和无包装它就是个月饼,里面的馅还是那个馅。
扯的有点远了。今天先写到这里发出来吧。下次再补充,EDG炸了我看的很难受= =。
========================================================================================
2015/11/16 补充:
本次补充主要根据《The C Programming Language》
1 #include <stdio.h> 2 3 int main() 4 { 5 short x = 65535+65535;//-2 6 7 short x1 = 65535; //取余 但是可以为负数 8 9 short x2 = 65537; //取余后的最小非负整数 10 unsigned short y = 65537; //取余后的最小非负整数 11 12 printf("x1=%d x2=%d " , x1 , x2); 13 printf("y = %d ", y ); 14 return 0; 15 16 }
ps
当表达式中包含 u n s i g n e d 类 型 的 运 算 分 量 时 , 转 换 规 则 要 复 杂 一 些 。 主 要 问 题 是 , 在 有 符 号 值与无符号值之间的比较运算取决于机器,因为它们取决于各个整数类型的大小。例如,假定
i n t 对象占 1 6 位, l o n g 对象占 3 2 位,那么,- 1 L < 1 U ,这是因为 i n t 类型的- 1 U 被提升为 s i g n e d l o n g
类型;但- 1 L > 1 U L , 这 是 因 为 - 1 L 被提升为 u n s i g n e d l o n g 类 型 , 因 此 它 是 一 个 比 较 大 的 正 数 。
在进行赋值时也要进行类型转换, = 右边的值要转换成左边变量的类型,后者即赋值表达式 结果的类型。
如前所述,不管是否要进行符号扩展,字符值都要转换成整数值。
当把较长的整型数转换成较短的整型数或字符时,要把超出的高位部分丢掉。