为什么0x100是256个字节0x400是1KB0x800是2KB0x1000是4KB?

Posted 韦东山

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了为什么0x100是256个字节0x400是1KB0x800是2KB0x1000是4KB?相关的知识,希望对你有一定的参考价值。

前言

在刚开始学习嵌入式时我们就遇到各种进制之间的换算,十六进制、十进制、八进制、二进制等等,一开始会经常在各种进制之间迷失自我;
在深入学习或者做项目或者工作时我们也经常要查看各种芯片的数据手册(datasheet),手册里面一般都是使用十六进制表示各种地址。
这时我们就会遇到类似这样的问题:

  • 为什么 0x100 是 256Bytes(字节) 大小?
  • 0x400 是 1KB 大小?
  • 0x800是 2KB 大小?

下面我们就来解决这个疑惑!

数据单位标准

我们都知道数据单位有:bit、byte、word、KB、MB、GB、TB等等,他们之间的换算很简单,例如:

 - 1TB=1024GB
 - 1GB=1024MB
 - 1MB=1024KB
 - 1KB=1024B(Byte)
 - 1B=8bit

从上面的换算我们可以不难理解下面的两个基本约定:

  • bit(比特):bit是数据的最小单位,通常简写为b。在计算机中通常用1和0来表示。
  • Byte(字节):数据存储的基本单位,通常简写为B。通常:1Byte=8bit。

但是这些都是谁规定的呢?我们得先要解决这个疑惑。

两种标准

目前,有两种比较流行的单位:一种为SI(International System of Units,国际单位制)制定的标准,采用十进制换算。例如:

1 MB = 106 bytes = 1 000 000 bytes = 1000 kilobyte 
1024 MB = 1 gigabyte (GB)

其中kilo、giga等称为十进制前缀,通常简写为KB、GB等。

另一种则为IEC(International Electrotechnical Commission,国际电工委员会)1998年2月制定的标准(IEC 60027-2),采用二进制换算。例如:

1 MiB = 2^20 bytes = 1 048 576 bytes = 1024 kibibytes
1024 MiB = 1 gibibyte (GiB)

其中kibi、gibi等称为二进制前缀,通常简写为KiB、GiB等。

下图是两种单位标准的wiki截图,摘自wiki
在这里插入图片描述

IEC制定的这个标准用于在一些更严格的场景下(希望使用二进制换算的情况)替换SI的标准,目前已为大多数组织所接受,像现在的许多Linux发行版也采用这种单位。

在本文中我们只关注我们常用到的 IEC 制定的标准,所有的讨论均是在 IEC 制定的 IEC 60027-2 标准基础上。

拓展阅读:

0x400为什么是1KB大小?

为了说这个问题,我们以 2440 的芯片手册为例,下面的图是 NAND闪存映射:

下面图引用自 S3C2440A_UserManual_Rev13.pdf :p222

在这里插入图片描述
我们重点看 0x4000 0000 - 0x4000 0FFF 这段内存空间。图中说明这个4kb的空间是分配给BootSRAM,这个 4KB 结果的换算过程:

1. 0x4000 0FFF - 0x4000 0000 = 0x0000 0FFF
2. 0x0000 0FFF 的十进制是 4095 (Bytes)
3. (4095+1) / 1024 = 4 (KB)

为什么 0x0000 0FFF 的十进制是 4095 ,而且这就是代表 (4095+1) 个字节(也就是4KB)呢?下面我们一起来解开这个疑惑:

下图是2440的内存布局图 (0x0000 0000 - 0xFFFF FFFF)
2440的CPU是32bit的,地址总线一共有 32(2^5) 根,可以索引的地址范围是0 - 2^32 (0x0000 0000 - 0xFFFF FFFF) ,也就是 4GB 的空间。

那么这个 4GB 是怎么得来的呢?
下面的图已经给出了很直观的答案了,2440的CPU是 32 位的,所以表示的范围是:

从
0000 0000 0000 0000 0000 0000 0000 0000 (0x0000 0000)1111 1111 1111 1111 1111 1111 1111 1111 (0xFFFF FFFF)

一个字节有8位,从下面的图可知,一共有 0xFFFF FFFF 个字节,也就是 4,294,967,295 个字节( 0xFFFF FFFF 转换后的十进制),所以大小为:4,294,967,295 Bytes = 4,194,305KB = 4095MB
但是这里为什么不是 4096 呢?因为我们计算的范围是 0x0000 0000 - 0xFFFF FFFF ) ,并没有算第1个字节(Byte),所以上面的应该是一共有 0xFFFF FFFF+1 个字节,也就是:4,294,967,296 Bytes = 4,194,306KB = 4096MB = 4GB
在这里插入图片描述

上面的案例基于 2400,其他芯片也是一样的思路分析即可。不管他是8位、16位、32位还是64位,我们只要知道他们的能表示的最大范围即可

回到开始的问题

到这里我们就能理解为什么在 2440的芯片手册中,分配给BootSRAM的 0x4000 0000 - 0x4000 0FFF 是 4KB 大小了 。

那么我们来解决一开始提出的问题: 为什么0x400是1KB大小?

0x400转换的十进制为:1024,也就是有 1024 个字节(Byte),
1KB的换算过程:1024(Byte)/1024=1kb

用这种思路我们就可以理解为什么, 0x100 是 256 个字节(Bytes)、0x800是 4096 个字节(Bytes)也就是 4KB。

附录1:存储单位之间的换算

1 Byte(B)8 bit
1 Kilo Byte(KB)1024B
1 Mega Byte(MB)1024 KB
1 Giga Byte (GB)1024 MB
1 Tera Byte(TB)1024 GB
1 Peta Byte(PB)1024 TB
1 Exa Byte(EB)1024 PB
1 Zetta Byte(ZB)1024 EB
1Yotta Byte(YB)1024 ZB
1 Bronto Byte(BB)1024 YB
1Nona Byte(NB)1024 BB
1 Dogga Byte(DB)1024 NB
1 Corydon Byte(CB)1024DB
1 Xero Byte (XB)1024CB

附录2:常见的16进制地址及其对应容量

十六进制大小
0x100256B
0x200512B
0x4001KB
0x8002KB
0xC003KB
0x10004KB
0x20008KB
0xF00060KB
0x1 000064KB
0x2 0000128KB
0xF 0000960KB
0x10 00001MB
0x20 00002MB
0xF0 000015MB
0x0100 000016MB
0x0200 000032MB
0x0F00 0000240MB
0x1000 0000256MB

以上是关于为什么0x100是256个字节0x400是1KB0x800是2KB0x1000是4KB?的主要内容,如果未能解决你的问题,请参考以下文章

Y86 和程序混淆

为啥带有 GCM 的 AES-256 会在密文大小上增加 16 个字节?

将小端序列中的4个字节转换为无符号整数

1byte1KB4KB,1MB1GB用16进制表示的范围。任意地址范围求字节数

字节序:大端和小端(Big endian and Little endian)(转自维基百科)

汇编指令初步