大小端字节序存储和字节序判断

本文详细解释了整数在内存中的不同表示方法(原码、反码和补码),介绍了大小端存储的概念,并通过C语言示例展示了如何判断字节序。讨论了大小端存储在不同处理器架构如X86、KEILC51、ARM和DSP中的应用。
摘要由CSDN通过智能技术生成

一、整数在内存中的储存

  整数的2进制表⽰⽅法有三种,即 原码、反码和补码有符号的整数,三种表⽰⽅法均有符号位和数值位两部分,符号位都是⽤0表⽰“正”,⽤1表⽰“负”,最⾼位的⼀位是被当做符号位,剩余的都是数值位。

正整数的原、反、补码都相同。
负整数的三种表⽰⽅法各不相同。

原码:直接将数值按照正负数的形式翻译成⼆进制得到的就是原码。
反码:将原码的符号位不变,其他位依次按位取反就可以得到反码。
补码:反码+1就得到补码。

对于整形来说:数据存放内存中其实存放的是补码。

二、什么是大小端

  其实超过⼀个字节的数据在内存中存储的时候,就有存储顺序的问题,按照不同的存储顺序,我们分为⼤端字节序存储和⼩端字节序存储,下⾯是具体的概念:
⼤端(存储)模式:
是指数据的低位字节内容保存在内存的⾼地址处,⽽数据的⾼位字节内容,保存在内存的低地址处
⼩端(存储)模式:
是指数据的低位字节内容保存在内存的低地址处,⽽数据的⾼位字节内容,保存在内存的⾼地址处。

 1.int main()
 2.{
 3.	int a = 0x11223344;
 4.	
 5.	return 0;
 6.}

在这里插入图片描述
  在我们调试时发现系统将十六进制数字0x11223344的低位字节 ”44“ 放到了内存的低地址处,所以这段代码字节序的存储方式是小端存储。

三、字节序判断

#include <stdio.h>
int check_sys()
{
	int i = 1;
	return (*(char *)&i);
}
int main()
{
	int ret = check_sys();
	if(ret == 1)
	{
		printf("⼩端\n");
	}
	else
	{
		printf("⼤端\n");
	}
	return 0;
}
小端

  这里我们设计了一个函数 “ check_sys ”,用它得到我们存储是大端还是小端的信息。

在这里插入图片描述

  我们知道数字“ 1 "d的十六进制为0x00000001

请添加图片描述
  当我们获取 “ i ” 的地址时,得到的是它低字节位的地址。然后我们将它强转为 “ char ”类型的数据,解引用后返回它的值,当主函数中变量 “ ret ”接收到函数返回值时,如果为1,就打印小端,如果不为1,就打印大端。
  因为我们对char类型的地址解引用只会得到它的低地址的数据,如果时小端存储,得到的数据就为01,反之大端存储得到的数据为00,借此我们就可以轻松判断字节序的存储顺序了。

四、为什么会有大小端

  这是因为在计算机系统中,我们是以字节为单位的,每个地址单元都对应着⼀个字节,⼀个字节为8 bit位,但是在C语⾔中除了8bit的 char 之外,还有16bit的 short 型,32bit的 long 型(要看具体的编译器),另外,对于位数⼤于8位的处理器,例如16位或者32位的处理器,由于寄存器宽度⼤于⼀个字节,那么必然存在着⼀个如何将多个字节安排的问题。因此就导致了⼤端存储模式和⼩端存储模式。
  例如:⼀个 16bit 的 short 型 x ,在内存中的地址为 0x0010 , x 的值为 0x1122 ,那么0x11 为⾼字节, 0x22 为低字节。对于⼤端模式,就将 0x11 放在低地址中,即 0x0010 中,0x22 放在⾼地址中,即 0x0011 中。⼩端模式,刚好相反。我们常⽤的 X86 结构是⼩端模式,⽽KEIL C51 则为⼤端模式。很多的ARM,DSP都为⼩端模式。有些ARM处理器还可以由硬件来选择是⼤端模式还是⼩端模式。

  • 25
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值