C语言数据类型专题之整型数值的转换:切片(截取)

本文详细介绍了整型变量在不同数据类型间的转换,特别是切片操作,以及在计算过程中的类型提升。通过实例分析了有符号和无符号整型在计算和赋值时的位宽扩展和截取规则,并解释了在比较操作中可能出现的类型转换问题。同时,讨论了C语言中算术类型转换的优先级。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

整型的不同类型数值之间的转换二:切片(将表示范围大的整型变量赋值给表示范围小的整型变量)

1. 切片指:从低地址到高地址截取相应的位!(由于数据在内存中是小端存储),仍然是补码运算!!

例如:int x; short a=x; //将int类型的4字节32位的变量赋值给short短整型2字节16位的变量

若x表示为:(高地址)0001 0010 0011 0100 0101 0110 1010 1000(低地址),则a为x截取(切片)低16位:0101 0110 1010 1000

2. 分析下列代码:

#include<stdio.h>
int main() {
	char a = 100;
	char b = 200;
	char c = a + b;
	printf("%d %d\n", c, a + b);//44 44
	unsigned char x = 100;
	unsigned char y = 200;
	unsigned char z = x + y;
	printf("%d %d", z, x + y);//44 300
	return 0;
}

注意是以%d的形式输出结果,意味着要以10进制整数(int类型,4字节,32位)形式输出。且c=a+b; 和z=x+y; 应该先计算,再扩充。

第一部分:

a,b,c均为char类型1字节有符号数。a=100=0110 0100,b==200=1100 1000,c=a+b=0110 0100+1100 1000=1 0010 1100,由于c为char类型1字节8bit位的数,故应截取(切片)低八位:c=0010 1100

输出结果时:c=0010 1100,有符号数扩充符号位,c=0000 0000 0000 0000 0000 0000 0010 1100=44。此时需要输出a+b:a(char类型,有符号数,1字节)=0110 0100,将其扩充为4字节为:a=0000 0000 0000 0000 0000 0000 0110 0100,同理b扩充后:b=1111 1111 1111 1111 1111 1111 1100 1000,故相加运算后a+b=1 0000 0000 0000 0000 0000 0000 0010 1100,截取低32位,结果为44。

第二部分:

x,y,z均为unsigned char类型1字节无符号数。x=100=0110 0100,y=200=1100 1000,z=x+y=0110 0100+1100 1000=1 0010 1100,截取低八位结果为:z=0010 1100

输出结果时:z=0010 1100,无符号数扩充0,z=0000 0000 0000 0000 0000 0000 0010 1100=44。对于x+y:x(无符号char类型)=0110 0100,将其扩充为4字节为:x=0000 0000 0000 0000 0000 0000 0110 0100,y(无符号char类型)=1100 1000,对其无符号数扩充0结果为:y=0000 0000 0000 0000 0000 0000 1100 1000,故相加运算后x+y=0000 0000 0000 0000 0000 0001 0010 1100=300

3. 关于算术类型的转换和赋值类型的转换

优先转换级别从高到低:double > unsigned long long > long long > unsigned int > int

下面的代码为什么会输出-5 > 10? 

#include<stdio.h>
int main() {
	char ch = -5;
	unsigned int x = 10;
	if (ch > x)	printf("%d>%d\n", ch, x);
	else printf("%d>%d\n", x, ch);
	return 0;
}

其实是由于在进行比较的时候,会自动发生类型的转换(类型提升~扩充):char会向unsigned int靠齐。char ch=-5=1000 0101(原码)=1111 1011(补码),由于其自身为有符号char类型1字节,要转为无符号int类型4字节,故其扩充符号位:ch=1111 1111 1111 1111 1111 1111 1111 1011(补码)=1000 0000 0000 0000 0000 0000 0000 0101(原码),由于转换为无符号类型,首位为数值位,故其为一个超级大的数!远远大于10=0000 0000 0000 0000 0000 0000 0000 1010。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值