基本上,如果我在 D1 中有一个数字,并且希望它始终是偶数,我将如何确保它永远不会是奇数?
我知道这与 AND 指令有关。但是当我尝试这样做时,它总是会减去 1。所以它会将奇数变为偶数,将偶数变为奇数。
基本上我该怎么做if n is odd, sub 1
基本上,如果我在 D1 中有一个数字,并且希望它始终是偶数,我将如何确保它永远不会是奇数?
我知道这与 AND 指令有关。但是当我尝试这样做时,它总是会减去 1。所以它会将奇数变为偶数,将偶数变为奇数。
基本上我该怎么做if n is odd, sub 1
and
你的号码与-2
。
在 2 的补码表示中,-2 是一个除最低位 (11111...110) 之外的所有位都设置为 1 的数字,因此,用作掩码时,它总是只删除数字的低位。这迫使它是偶数(即使对于负数它也能正常工作)。
至于标题中的“viceversa”:做相反的事情(=强制每个偶数到下一个奇数),or
只需1
. 这会将低位设置为 1,从而获得所需的效果。
奇数1
以二进制结尾,偶数以0
二进制结尾。您真正想要的是使最后一个二进制数字成为0
,无论它以什么开头。(这将从奇数中减去 1,并保持偶数不变。)
做到这一点的方法是与 与1111...1110
,其中所有二进制数字都不1
是最后一个数字,即0
. 您可以通过对 进行按位否定来构造它0000...0001
,这当然是1
.
所以如果你的数字是n
,你要计算n & (~1)
。
68000 CPU 实际上有单独的位设置和位清除命令。在像您这样的情况下,显式清除操作数的位 0 而不恢复为布尔逻辑可能更简单、更清晰:
bclr.l #0,d0
代替
and.l $fffffffe,d0
你只需要清除底部位,所以当其他人使用长字指令时,使用 AND 的字节形式并没有错
AND.B #0xFE, D1 ; make even
OR.B #0x01, D1 ; make odd
XOR.B #0x01, D1 ; toggle even/odd
无论 D1 中值的大小(字节、字或长字),您仍然清除了底部位并保持寄存器中的所有其他位不变
或者
BCLR #0,D1 ; make even
BSET #0,D1 ; make odd
BCHG #0,D1 ; toggle even/odd
对于 16 位数字,请执行以下操作:
EvenNumber = (D1 & 0xFFFE);