整型的不同类型数值之间的转换二:切片(将表示范围大的整型变量赋值给表示范围小的整型变量)
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。