深入整形数据在内存中的存储

整形家族(在VS编译器中char默认为singned,其他编译器中不确定 ,其他整形单作有符号处理列如(int ==singned int)

char

unsigned char

无符号字符 unsigned char 范围(0~255)

00000000~11111111 (有效数值位有八位)

signed char 范围(-128~127)

00000000 0

00000001 1

............

01111111 127

10000000 -128

10000001 -127

................

11111111 -1

*********************************************************************

short

unsigned short [int] 范围(0~65535)

00000000 00000000 ~11111111 11111111

signed short [int] 范围(-32768~32767)

00000000 00000000 0

00000000 00000001 1

.............................

01111111 11111111 32767

10000000 00000000 -32768

...............................

11111111 11111111 -1

**********************************************************************

int

unsigned int 范围(0~2^32)

signed int

int 分为有符号int singned int (一般 int==singned int )

无符号整形 (最高位符号位为有效位,有32位) unsigened int

00000000 00000000 00000000 00000000 00000000 ~ 111111111 11111111 11111111 11111111 11111111 范围为 0~2^32(4294967296)

long

unsigned long [int] 范围(0~2^32)

signed long [int]

一.

正整数的补码,原码,反码是 相同的(这是规定)

int a=5;

原码 000000000000000000000000000000101

反码 11111111111111111111111111111111111010

补码 11111111111111111111111111111111111001

int 类型存储为32位,最高位为符号位,0表示正数,1表示负数

int a=-5;

负数的反码是原码的取反,补码为反码+1,位移结果必须要计算

原码 100000000000000000000000000000101

反码 11111111111111111111111111111111111010

补码 11111111111111111111111111111111111001

二.计算机中在内存中存储数据的顺序分大小端

1.大端存储:数据的低位放在高地址,高位放在低地址

2.小端存储:数据的低位放在低地址,高位放在高地址

以int a=0x11223344为例子,这里我们用16进制表示a,二进制表示为十六进制,四个比特位为一个十六进制位,计算机中地址是以一个字节为单位计算地址的,八个比特位表示为一个字节 所以11 22 33 44分别表示四个字节

0x11223344 高位

这里我们从程序内存中看到 内存低地址存储数据的低位 44 所以这里是小端存储

0x000000C5376FFB84 44

0x000000C5376FFB85 33

0x000000C5376FFB86 22

0x000000C5376FFB87 11

这里我们用程序判断大小端,用char* 访问一个字节就可以判断

int test1(int n)//写法一:函数
{
return *(char*)&n;
}
int main()
{
//写法二
int a = 1;//00000000 00000000 00000000 00000001      高  <-----------低
char* p = (char*)&a;
if (*p == 1)
{
printf("小端");
}
else
{
 printf("小端");
}
//写法三
int b = 2;//00000000 00000000 00000000 00000010
if (*(char*)&b)
{
    printf("小端");

}
else
{
    printf("大端");
}
//调用函数test1
int ret = test1(a);
if (ret == 1)
    printf("小端");
else
{
    printf("大端");
}
return 0;

}

三 ,数据在内存中的存储行为

//#include<stdio.h>
十进制打印字符整形
//int main()
//{
//
//    char  a = 1;
//    //000000000 00000000 00000000 00000001
//    //000000000 00000000 00000000 00000001
//    //000000000 00000000 00000000 00000001
//    //截断   00000001 (因为char类型只分配一个字节大小空间所以对于整形发生截断只拿取最后的一个字节)      1
//    char   b = -1;
//    //10000000  00000000 00000000 00000001
//    //11111111  11111111 11111111 11111110
//    //11111111  11111111 11111111 11111111
//    //截断     11111111
//    //  补码 11111111
//    //  反码 11111110
//    //  原码 10000001    b=-1
//
//    signed char c = -1;
//    // 截断 11111111
//    unsigned  char d = -1;
//    //截断  11111111
//
//
//    printf("%d %d %d %d ", a, b, c, d);//发生整形提升,无符号位补0,有符号看类型
//
//    // a  补码 00000000 00000000 00000000 00000001 正数 补码=原码=反码   a=1;
//    //b   补码 11111111 11111111 11111111 11111111  负数  原码为 10000000 00000000 00000000 00000001  b=-1;
//    //c   补码 11111111 11111111 11111111 11111111  负数  原码为 10000000 00000000 00000000 00000001  c=-1;
//    //d   补码 00000000 00000000 00000000 11111111  正数  补码=原码=反码   d=255
//    return 0;
//}
//#include<stdio.h>
//#include<windows.h>
//int main()
//{
//    char a = -128;
//    //10000000 00000000 00000000 10000000
//    //11111111 11111111 11111111 01111111
//    //11111111 11111111 11111111 10000000
//    //截断 10000000
//    //以十进制打印所以进行整形提升  补码   11111111 11111111 11111111 10000000
        //因为要打印无符号,补码==原码   
//    printf("%u\n", a);

//
//
//    char b = 128;
//    //00000000 00000000 00000000 10000000
//    //截断   补码 10000000
//    //整形提升 补码 11111111 11111111 11111111 10000000
//    //
//    printf("%u\n", b);
//
//       这里我们发现这里a和b 的值相同,因为截断后是相同的
//      **************************************************
//#include<stdio.h>
int main()
{
    int j =10;
    //00000000 00000000 00000000 00001010
    
    unsigned i = -20;
    //10000000 00000000 00000000 00010100
    //11111111 11111111 11111111 11111111
    //11111111 11111111 11111111 11101100

    //i+j 补码相加
    // i的补码 00000000 00000000 00000000 00001010
    // j的补码 11111111 11111111 11111111 11101100
    //i+j的补码11111111 11111111 11111111 11110110
    //i+j的原码10000000 00000000 00000000 00001001   值为-10
    printf("%d\n", j + i);
    //这里打印无符号整形,因为i的字节大小已经是整形了,不发生整形提升,无符号打印原码==补码,值为4294967276
    printf("%u", i);

  return  0;


}

四 ,计算机处理整形溢出

(在平时做题如果我们不考虑一个整形大小,有时存储的值会大于整形值的大小就会发生溢出)

这里我们以有符号char 为列子

#include<stdio.h>
int main()
{
    signed char a = 127;
    a += 1;
    printf("%d", a);


    return 0;
}

我们来解释一下

signed char 范围(-128~127)

00000000 0

00000001 1

............

01111111 127

10000000 -128

10000001 -127

................

11111111 -1

如果我们存储的值为127+1 就超出范围了

127    01111111 + 1 逢二进一 得到 10000000 转换成原码 就是 -128

不断加一 ,就会一直在 -128~127中循环

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值