混合运算中不同数据类型之间的隐式转换原则(C语言)
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 型。式中右面表达式按如下步骤处理:
- 首先将'a'、'b'和s换成 int ,将1.5和f转换为 double 型。
- 计算100+'a',因'a'已转换为 int 型,于是此运算结果为197。
- 计算1.5*u,由于1.5已转换为 double ,u是 unsigned 型,于是首先u转换为 double ,然后进行运算,运算结果为 double 。
- 计算197+1.5 * u,先将197转换为 double (如197.00…00),其结果为 double 。
- 计算f/ 'b',f已转换为 double ,'b'已转换为 int ,于是先将'b'再转换为 double ,其结果为 double 。
- 计算(197+1.5 * u)+f / 'b',者均为 double ,于是结果也为 double 。
- 计算s * 3.1415926,先将s由int转换为 double ,然后进行运算,其结果为 double 。
- 之后与前面得的结果相减,结果为 double 。
- 最后将表达式的结果转换为 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
规则总结
范例解析
unsigned int i = 3; cout << i * -1; // 输出?
各数据类型的表示范围
来自MSDN
补充说明:
答案
(整理自网络)
参考资料:
https://blog.csdn.net/zhuimengzh/article/details/6728492
https://blog.csdn.net/miaouu/article/details/5213042