c语言中的类型转换

    首先引出一个问题,这是我在编写模拟器程序的时候遇到的,我们有如下的类型转换:

       int16_t     v16s;
       uint16_t   v16u;
       int32_t     v32s;
       uint32_t   v32u;

       v16s = 0xf08b;     v16u = (uint16_t)v16s; //请问此时v16s和v16u在二进制表示形式上有什么不同吗? No.1
       v32s = (int32_t)v16s;      v32u = (uint32_t)v16s; //请问此时v32s和v32u在二进制表示形式上相同吗? No.2
       v16s = 0x0f8b;  v32s=(int32_t)v16s;   v32u=(uint32_t)v16s; //请问此时v32s和v32u的值又是多少? No.3
       v32s = 0xffff0fb8; v16s=(int16_t)v32s; v16u=(uint16_t)v32s;//请问此时v16s和v16u的值是多少? No.4
       v16s = 0xf08b; v16u=0xf08b;  v32s=(int32_t)v16s; v32u=(uint32_t)v16u; //请问此时v32s和v32u的值是多少? No.5

首先告诉大家答案:

  1. No.1:请问此时v16s和v16u在二进制表示形式上有什么不同吗?
    二者在二进制表示形式上没有区别,都是0xf08b,所以如果不进行运算操作,在同样长度的简单数据类型之间进行强制类型转换是不必要的,编译器会略去你的工作,不信的话,你可以查看生成的汇编代码,对于No.1问题对转换语句,根本就不生成代码!!
  2. No2:请问此时v32s和v32u在二进制表示形式上相同吗?
    二者在二进制表示形式上是相同的!都是0xfffff08b!这里可能会犯错的地方在 v32u=(uint32_t)v16s这个语句上,我就是这么犯的错误。按照我错误的理解是:0xf08b → 0x0000f08b!这是不对的!原因后面再告诉你(当然如果你C语言学的够好,就觉得我再弄小儿科了^_^,照顾一下后进者)。
  3. No.3:请问此时v32s和v32u的值又是多少?
    此时v32s和v32u的值都是 0x00000f8b!
  4. No.4:请问此时v16s和v16u的值是多少?
    此时v16s和v16u的值都是0x0fb8!
  5. No.5:此时v32s的值是0xfffff08b,v32u的值是0x0000f08b。在进行自动类型转换的时候,如果原来的数是无符号数,那么在扩展的时候,高位填充的是0;如果是有符号数,那么高位填充的时符号位!这一点有点类似于“>>”操作符,当无符号数右移的时候,高位填充的是0;有符号数右移的时候,高位填充的是符号位。
    好了,下面我就来告诉你,为什么是上面的结果。这就需要我们回头复习一下C语言的有关类型转换的操作。C语言的类型转换,可以分为两种: 自动类型转换(隐式类型转换,有编译器帮你去完成)和 强制类型转换(你知道自己想要什么,所以才转换)。
    对于 自动类型转换,最常见的就是混合运算以及赋值运算,还有一种就是函数值的类型转换
  1. 赋值运算:自动把“=”右边的表达式的类型转换成“=”右边的变量的类型,例如 int a=4.5; a的值实际是4!
  2. 混合运算:就是一个运算表达式当中包含了多个类型,这时候就需要有类型转换。当运算符两边的操作数类型不同时,其中一个操作数就要经过类型转换以和另一个操作数的类型相一致,然后才能进行运算。
    变换操作数采取就高不就低的原则,即级别低的操作数先被转换成和级别高的操作数具有同一类型,然后再进行运算,结果的数据类型和级别高的操作数相同。

           高        double    ←←    float
           ↑          ↑             
           ↑         long     
           ↑          ↑
           ↑        unsigned
           ↑          ↑
           低         int
          ←←    char,short

                 自动转换顺序表

  3. 函数返回值的类型转换: int f1(){ return 36.8;}
    强制类型转换运算符
    可以利用强制类型转换运算符将一个表达式转换成所需类型:
    例如:
  (double)a       (将a转换成double类型)
     (int)(x+y)      (将x+y的值转换成整型)
     (float)(5%3)    (将5%3的值转换成float型)
     (int)(1.5+2.3) = ? 
      (int)1.5+2.3=?
    注意,表达式应该用括号括起来。如果写成(int)x+y 则只将x转换成整型,然后与y相加。

    讲到这里,您或许就明白了,哦,原来我们是使用了强制类型转换的语法,但是程序实质上却是自动类型转换! 所以在执行了语句:v16s = 0xf08b;  v32s=(int32_t)v16s;    v32u=(uint32_t)v16s; 后v32s和v32u的二进制表示形式才是一样的!
    对于从高到低的强制转换,实质上就是一个截断的操作,只把低端需要的部分保留,其余的部分直接扔掉了。所以对于语句:
v32s = 0xffff0fb8; v16s=(int16_t)v32s; v16u=(uint16_t)v32s; 执行之后v16s和v16u的二进制表示形式才是一样的!
    注:我一直强调的都是它们的二进制表示形式是否相同,而没有说它们的值是否相同,因为同样的二进制表示形式,你把它当作有符号数和无符号数的值在决大部分的情况下都是不同的(除了最高位为0的相同外,其他都不同)

    混合运算

 混合运算是指在一个表达式中参与运算的对象不是相同的数据类型,例如:
  2*3.1416*r 3.1416*r*r 3.6*a%5/(*b)+'f';
 如果r为int型变量,a为float型变量,b为double型变量,则以上三个表达式中涉及到的数据类型有整型、实型、字符型,这种表达式称为混合类型表达式。对混合类型表达式的求解要进行混合运算,此时首要的问题是对参与运算的数据进行类型转换。

下面给出类型转换的示例,以加深理解。设有如下变量说明:
  int a, j, y; float b; long d; double c;
  则对赋值语句:
  y=j+'a'+a*b-c/d;
  其运算次序和隐含的类型转换如下:
  ① 计算a*b,由于变量b为float型,所以运算时先由系统自动转换为double型,变量a为int型,两个运算对象要保持类型一致,变量a也要转换为double,运算结果为double型。
  ② 由于c为double型,将 d 转换成 double 型,再计算 c/d,结果为double型。
  ③ 计算j+'a',先将'a'(char型)转换成整型数再与j相加,结果为整型。
  ④ 将第1步和第3步的结果相加,先将第3步的结果(int)转换成double型再进行运算,结果为double型。
  ⑤ 用第4步的结果减第2步的结果,结果为double型。
  ⑥ 给y赋值,先将第5步的结果double型转换为整型(因为赋值运算左边变量y为整型),即将double型数据的小数部分截掉,压缩成int型,然后进行赋值。
  以上步骤中的类型转换都是C语言编译系统自动完成的。
  • 5
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值