C语言的类型提升机制

概念

在C语言中,整数类型按照其大小可以分为以下几类(从小到大):

  • char
  • short
  • int
  • long
  • long long

当在表达式中涉及这些类型的混合运算时,较小的类型会被提升为较大的类型。具体规则如下:

①charshort 在大多数情况下会被提升为 int

②如果 int 可以表示 charshort 的所有值,则 charshort 会被提升为 int

③如果 int 不能表示 charshort 的所有值(这种情况很少见,通常只在某些特殊硬件上发生),则 charshort 会被提升为 unsigned int

④intunsigned int 在运算时不会被提升为更大的类型,除非与更大的类型(如 long)混合使用。

⑤longunsigned long 在运算时通常不会被提升为 long long,除非与 long long 混合使用。

示例

来看这样的一段代码来了解这个机制的具体作用:

cpp 复制代码
#include <stdio.h>
#include <stdint.h>

int main()
{
	int8_t arr[4];
	uint8_t arr1[4];

	arr[0] = 0x00;
	int8_t x=~arr[0];
	if(x==~arr[0])
	{
		printf("int8_t x=%d\n", x);
	}else{}
	
	arr1[0] = 0x00;
	uint8_t y = ~arr1[0];


	if (y==~arr1[0])
	{
		printf("Hello World\n");
	}else{}

	return 0;
}

编译运行的输出结果为:

仅仅输出了第一个if语句内的内容,原因是什么呢?可以试着把第二个if语句内要输出的内容改成"hello world"试试,结果依然是只输出了第一个,初步推测是第二个if语句内的条件不为真导致的。

调试

按下F10开始调试,点击调试->窗口->监视;

然后设置三个观察条件,分别是arr1[0]、y以及~arr1[0];

在最后一个空的else代码块打断点,并且运行到它的if语句那一行;

可以看到和Keil一样,有断点(程序暂停)和箭头的UI(程序当前运行停止的位置)。

从监视栏也可以看到~arr1[0]的类型从unsigned char类型被提升为了int型,并且此时y=255不等于~arr1[0]。

我们再直接输出结果看看:

可以看到如果强行及将~arr[0]作为unsigned char类型输出,结果是0xFFFFFFFF。

注意数据在一切计算机(单片机、Soc、PC等)中的存储形式都是补码,Visual Stdio中我们选择了32位的数据处理位宽那么此时~arr[0]的实际内容就是1111 11111 1111 1111 1111 1111 1111 1111。

更改一下占位符:

作为int类型输出,结果就是-1(-1的补码是32个1)。

注意

C语言的强弱类型之争到今天也没结束,类型提升导致的bug是可以通过良好的编码习惯避免的,我们在日常编程的过程中要尽量避免不同类型的数据之间进行数值比较。

相关推荐
飞机和胖和黄1 分钟前
王道C语言第一周作业
c语言·开发语言
lly2024062 分钟前
SQLite 安装指南
开发语言
星火开发设计4 分钟前
C++ deque 全面解析与实战指南
java·开发语言·数据结构·c++·学习·知识
ID_1800790547311 分钟前
除了Python,还有哪些语言可以解析淘宝商品详情API返回的JSON数据?
开发语言·python·json
草莓熊Lotso13 分钟前
Qt 信号与槽深度解析:从基础用法到高级实战(含 Lambda 表达式)
java·运维·开发语言·c++·人工智能·qt·数据挖掘
superman超哥1 小时前
Rust 异步错误处理最佳实践
开发语言·rust·编程语言·rust异步错误处理·rust最佳实践
脏脏a1 小时前
C++ STL list 模拟实现:从底层链表到容器封装
开发语言·c++·stl·双链表
故事不长丨8 小时前
C#正则表达式完全攻略:从基础到实战的全场景应用指南
开发语言·正则表达式·c#·regex
哈库纳玛塔塔9 小时前
放弃 MyBatis,拥抱新一代 Java 数据访问库
java·开发语言·数据库·mybatis·orm·dbvisitor
你怎么知道我是队长9 小时前
C语言---typedef
c语言·c++·算法