首页 > 编程语言 > 详细

混合运算中不同数据类型之间的转换原则(C语言)

时间:2020-12-10 21:46:56      阅读:145      评论:0      收藏:0      [点我收藏+]

C语言的数据类型

c语言中,有关数据类型的分类,网上有很多种版本,未找到比较统一的标准,下图为我比较认可的划分:

技术分享图片

常量的默认类型

在C语言中,数值常量分为两种:整数和小数。整数常量默认为有符号的int类型,而小数常量默认为double类型。字符常量被认为是一种特殊的整型常量,默认类型为有符号的char,而字符串常量一般被认为const char类型。需要注意的是,不同的编译器可能略有不同,但一般遵守该规则。

隐式转换

C语言在以下四种情况中会进行隐式转换:
(1)算术运算式中,类型能够换为类型。

    所谓高低,是指表示的范围大小。通常和内存默认分配的存储空间大小及有无符号位有关。

技术分享图片

(2)赋值表达式中,右边表达式的值自动隐式换为左边变量的类型,并赋值给它。
(3)函数调用中参数传递时,系统隐式地将实参转换为形参的类型后,赋给形参。
(4)函数有返回值时,系统将隐式地将返回表达式类型换为返回值类型,赋值给调用函数。

算数运算中的转换规则

算数运算中,有如下类型转换规则:

(1)字符必须先转换为整数(C语言规定字符类型数据和整型数据之间可以通用) 。
(2)short换为int型(同属于整型) 。
(3)float型数据在运算时一律换为双精度(double)型,以提高运算精度(同属于实型,即浮点型) 。

总的来说,当不同类型的数据进行操作时,应当首先将其转换成相同的数据类型,然后进行操作,转换规则是由低级向高级转换(如上图所示)。

算术运算示例

执行:x = 100 + ‘a‘ + 1.5 * u + f / ‘b‘ - s * 3.1415926

其中,u为unsigned型,f为float型,s为short型,x为float型。式中右面表达式按如下步骤处理:

  1. 首先将‘a‘、‘b‘和s换成int,将1.5和f转换为double型。
  2. 计算100+‘a‘,因‘a‘已转换为int型,于是此运算结果为197。
  3. 计算1.5*u,由于1.5已转换为double,u是unsigned型,于是首先u转换为double,然后进行运算,运算结果为double。
  4. 计算197+1.5 * u,先将197转换为double(如197.00…00),其结果为double。
  5. 计算f/ ‘b‘,f已转换为double,‘b‘已转换为int,于是先将‘b‘再转换为double,其结果为double。
  6. 计算(197+1.5 * u)+f / ‘b‘,者均为double,于是结果也为double。
  7. 计算s * 3.1415926,先将s由int转换为double,然后进行运算,其结果为double。
  8. 之后与前面得的结果相减,结果为double。
  9. 最后将表达式的结果转换为float并赋给x。

有符号数与无符号数之间运算问题

当表达式中存在有符号类型和无符号类型时,所有的操作数都自动转换为无符号类型。因此,从这个意义上讲,无符号数的运算优先级要高于有符号数,这一点对于应当频繁用到无符号数据类型的嵌入式系统来说是丰常重要的。 

首先进行一个实验,分别定义一个signed int型数据和unsigned int型数据,然后进行大小比较:

unsigned int a = 20;
signed int b = -130;

实验证明:$b > a$,也就是说$-130 > 20$,为什么会出现这样的结果呢?

这是因为在C语言操作中,如果遇到无符号数与有符号数之间的操作,编译器会自动转化为无符号数来进行处理,因此$a=20, b=4294967166$,这样比较下去当然$b > a$了。再举一个例子:

 unsigned int a = 20;
 signed int b = -130;
 std::cout << a + b << std::endl;

结果输出为$4294967186$。同样的道理:在运算之前,$a=20$,$b$被转化为$4294967166$,所以$a+b=42949671864,减法和乘法的运算结果类似。

如果作为signed int型数据的b=-130,b与立即数之间操作时不影响b的类型,运算结果仍然为signed int型:

signed int b = -130;
std::cout << b + 30 << std::endl;
// 输出为-100

而对于浮点数来说,浮点数(float,double)实际上都是有符号数,unsigned和signed前缀不能加在float和double之上,当然就不存在有符号数根无符号数之间转化的问题了。

 示例

#include <iostream>
/*
当表达式中存在符号类型和无符号类型时
所有的操作数都自动转换为无符号类型
*/
using namespace std;
char getChar(int x,int y){
    char c;
    unsigned int a = x;
    unsigned int b = a + y;
    (a + y > 10)?(c = 1):(c = 2);
    return c;
}
void main(){
char c1 = getChar(7,4);
char c2 = getChar(7,3);
char c3 = getChar(7,-7);
char c4 = getChar(7,-8);
 
printf("c1=%d\n",c1);
printf("c2=%d\n",c2);
printf("c3=%d\n",c3);
printf("c4=%d\n",c4);
system("pause");
}
// 输出: c1=1  c2=2  c3=2  c4=1

规则总结

1、所有比int型小的数据类型(包括char,signed char,unsigned char,short,signed short,unsigned short)转换为int型。如果转换后的数据会超出int型所能表示的范围的话,则转换为unsigned int型;
2、bool型转化为int型时,false转化为0,true转换为1;反过来所有的整数类型转化为bool时,0转化为false,其它非零值都转为true;
3、如果表达式中混有unsigned short和int型时,如果int型数据可以表示所有的unsigned short型的话,则将unsigned short类型的数据转换为int型,否则,unsigned short类型及int型都转换为unsigned int类型。举个例子,在32位机上,int是32位,范围–2,147,483,648 to 2,147,483,647,unsigned short是16位,范围0 to 65,535,这样int型的足够表示unsigned short类型的数据,因此在混有这两者的运算中,unsigned short类型数据被转换为int型;
4、unsigned int与long类型的转换规律同3,在32位机上,unsigned int是32位,范围0 to 4,294,967,295,long是32位,范围–2,147,483,648 to 2,147,483,647,可见long类型不够表示所有的unsigned int型,因此在混有unsigned int及long的表达式中,两者都被转换为unsigned long;
5、如果表达式中既有int 又有unsigned int,则所有的int数据都被转化为unsigned int类型。

范例解析

unsigned int i = 3;
cout << i * -1;
// 输出?

各数据类型的表示范围

来自MSDN

技术分享图片

补充说明:

1)在32位机上,int型和unsigned int型都是32位(4个字节)。
2)enum会跟据最大值来决定类型,一般来说为int型,如果超出int型所能表示的范围,则用比int型大的最小类型来表示(unsigned int, long或者unsigned long)。
3)关于类型的大小。一般用所能表示的数据范围来比较类型的大小,如char型<unsigned char型<short型...在表达式中,一般都是由小的类型向大的类型转换(强制类型转换除外)。

答案

输出是4294967293。
原因:在表达式$i * -1$中,$i$是unsigned int型,$-1$是int型(常量整数的类型同enum),按规则总结第5条可以知道$-1$必须转换为unsigned int型,即十六进制的0xFFFFFFFF,十进制的$4294967295$,然后再与$i$相乘,即$44294967295 * 3$,如果不考虑溢出的话,结果是$12884901885$,十六进制$0x2FFFFFFFD$,由于unsigned int只能表示32位,因此结果是0xFFFFFFFD,即十进制的$4294967293$。

(整理自网络)

参考资料:

https://blog.csdn.net/zhuimengzh/article/details/6728492

https://blog.csdn.net/miaouu/article/details/5213042

混合运算中不同数据类型之间的转换原则(C语言)

原文:https://www.cnblogs.com/MinPage/p/14117237.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!