通用寄存器主要有ax,bx,cx,dx,用来存储一般性的数据。8086CPU中的寄存器都是16位的,可以存放两个字节。寄存器中的位是寄存器的最小存储单元,它可以保存一个“0”或者一个“1”,也就是一个二进制数据。所以一个16位的寄存器所能存储的数据的范围是什么呢。在这个问题之前我想先提一下原码,反码和补码之间的关系。
1. 原码
原码就是符号位加上真值的绝对值, 即用第一位表示符号, 其余位表示值. 比如如果是8位二进制:
[+1]原 = 0000 0001
[-1]原 = 1000 0001
第一位是符号位. 因为第一位是符号位, 所以8位二进制数的取值范围就是:
[1111 1111 , 0111 1111]
即
[-127 , 127]
2. 反码
反码的表示方法是:
正数的反码是其本身
负数的反码是在其原码的基础上, 符号位不变,其余各个位取反.
[+1] = [00000001]原 = [00000001]反
[-1] = [10000001]原 = [11111110]反
3. 补码
补码的表示方法是:
正数的补码就是其本身
负数的补码是在其原码的基础上, 符号位不变, 其余各位取反, 最后+1. (即在反码的基础上+1)
[+1] = [00000001]原 = [00000001]反 = [00000001]补
[-1] = [10000001]原 = [11111110]反 = [11111111]补
为什么要设立补码呢?
第一是为了能让计算机执行减法:
[a-b]补=a补+(-b)补
第二个原因是为了统一正0和负0
正零:00000000
负零:10000000
这两个数其实都是0,但他们的原码却有不同的表示。
但是他们的补码是一样的,都是00000000
特别注意,如果+1之后有进位的,要一直往前进位,包括符号位!(这和反码是不同的!)
[10000000]补
=[10000000]反+1
=11111111+1
=(1)00000000
=00000000(最高位溢出了,符号位变成了0)
规定:10000000这个补码表示的事-128
所以n位补码表示的范围就是:-2^(n-1)-2^(n-1)-1
在计算机系统中,数值一律用补码来表示(存储)
(1)无符号整数的范围是0-2^16-1
因为寄存器中的位是寄存器的最小存储单元,它可以保存一个“0”或者一个“1”
所以16位寄存器能表示的最小无符号数是0000 0000 0000 0000=0
最大无符号数是1111 1111 1111 1111=2^16-1
(2)有符号整数的范围是-2^15-2^15-1
2^15-1:0111 1111 1111 1111=2^15-1 0是符号位,计算数据的大小是符号位是不考虑的
-2^15:1111 1111 1111 1111=-2^15
使用补码,可以将符号位和其它位统一处理;同时,减法也可按加法来处理。另外,两个用补码表示的数相加时,如果最高位(符号位)有进位,则进位被舍弃。
通用寄存器都可以作为两个独立的寄存器来使用,例如ax可以分为al和ah,al为低8位(0-7位),ah为高8位(8-15位)。
字在寄存器中的存储:字节(byte)有8个bit组成可以存在8位的寄存器中。
字(Word)由两个字节组成,这两个字节分别是高字节和低字节。 (个人觉得这个是要记牢的)
关于汇编指令的地方大家可以仔细去阅读书,我觉得书上讲的很详细。
物理地址=段地址*16+偏移地址(物理地址唯一,但是逻辑地址(段地址*16+偏移地址)并不唯一)
段地址并不是实际意义上的将内存地址分为一段一段的,这样只是为了我们可以用分段的方式来管理内存,更便于管理而已。
说到段地址不免要介绍一下段寄存器CS和IP了。
CS是指代码段寄存器存放段地址,IP是指指令指针寄存器存放偏移地址。但是要记住mov指令是不能修改CS和IP的值的。
以上的内容如果有不对的地方欢迎指正。