51單片機(jī)匯編編程:BCD 碼與 BIN 數(shù)據(jù)的轉(zhuǎn)換程序
掃描二維碼
隨時(shí)隨地手機(jī)看文章
把二進(jìn)制數(shù)轉(zhuǎn)換成 BCD 碼,是很常見(jiàn)的,做而論道以前也發(fā)表過(guò)這樣的程序。
但是,過(guò)去都是用寄存器,如果要求用片內(nèi)RAM單元,就少見(jiàn)了。
當(dāng)然,也可以把片內(nèi)RAM單元的數(shù)據(jù),先傳送到寄存器,再利用以前寫(xiě)的程序進(jìn)行轉(zhuǎn)換。轉(zhuǎn)換完畢后,再把結(jié)果傳送到片內(nèi)RAM單元。
如果直接用片內(nèi)RAM單元來(lái)進(jìn)行轉(zhuǎn)換呢?
這也是可以的,只是速度會(huì)慢一些。
以前,從來(lái)沒(méi)有見(jiàn)過(guò)有誰(shuí)編寫(xiě)出來(lái)直接使用片內(nèi)RAM轉(zhuǎn)換的程序。
沒(méi)有人編寫(xiě),大概就是速度的原因吧。
做而論道現(xiàn)在就編寫(xiě)一個(gè),大家看看,和使用寄存器相比,速度有多少降低。
下面就是這樣的題目。
===============================
編寫(xiě)一段程序,將存放于片內(nèi)RAM 40H、41H單元中的 16 位二進(jìn)制數(shù)轉(zhuǎn)化為壓縮BCD碼存放到 38H~3AH 中。
TO_BCD:
? ? MOV ? R2, #16 ? ? ;轉(zhuǎn)換16位
? ? CLR ? A
? ? MOV ? 38H, A ? ? ?;先清零
? ? MOV ? 39H, A
? ? MOV ? 3AH, A
LOOP:
? ? MOV ? A, 41H ? ? ?;取被轉(zhuǎn)換二進(jìn)制數(shù)
? ? RLC ? A
? ? MOV ? 41H, A
? ? MOV ? A, 40H
? ? RLC ? A ? ? ? ? ? ;最高位移動(dòng)到C
? ? MOV ? 40H, A
;---------------------
? ? MOV ? A, 3AH ? ? ?;取數(shù)
? ? ADDC ?A, 3AH ? ? ?;左移并加上C
? ? DA ? ?A ? ? ? ? ? ;轉(zhuǎn)為BCD碼
? ? MOV ? 3AH, A ? ? ?;保存
? ? MOV ? A, 39H
? ? ADDC ?A, 39H
? ? DA ? ?A
? ? MOV ? 39H, A
? ? MOV ? A, 38H
? ? ADDC ?A, 38H
? ? MOV ? 38H, A
? ? DJNZ ?R2, LOOP ? ?;循環(huán)16次
RET
本程序已經(jīng)經(jīng)過(guò)試驗(yàn),保證正確。
===============================
下面,是另外一個(gè)問(wèn)題,問(wèn)的就是相反轉(zhuǎn)換的問(wèn)題。
這樣的程序,做而論道以前就編寫(xiě)出來(lái),并且使用過(guò)很多年了。
只是在網(wǎng)上、書(shū)上,從來(lái)也沒(méi)有見(jiàn)過(guò)這樣的轉(zhuǎn)換程序。
----
編寫(xiě)一段程序,將存放于片內(nèi) RAM 30H~32H 單元中的5位壓縮BCD數(shù)(小于65536)轉(zhuǎn)化為二進(jìn)制數(shù)存放到 40H、41H 單元中。
TO_BIN:
? ? MOV ? A, 32H
? ? SWAP ?A
? ? ANL ? A, #0FH ? ? ;得到十位數(shù)字
? ? MOV ? B, #10
? ? MUL ? AB
? ? MOV ? B, 32H
? ? ANL ? B, #0FH ? ? ;得到個(gè)位數(shù)字
? ? ADD ? A, B
? ? MOV ? 41H, A ? ? ?;(41H)=十位*10+個(gè)位
;---------------------
? ? MOV ? A, 31H
? ? SWAP ?A
? ? ANL ? A, #0FH ? ? ;得到千位數(shù)字
? ? MOV ? B, #10
? ? MUL ? AB
? ? MOV ? B, 31H
? ? ANL ? B, #0FH ? ? ;得到百位數(shù)字
? ? ADD ? A, B ? ? ? ?;(A)=千位*10+百位
? ? MOV ? B, #100
? ? MUL ? AB ? ? ? ? ?;(B A)=千位*1000+百位*100
;---------------------
? ? ADD ? A, 41H
? ? MOV ? 41H, A
? ? CLR ? A
? ? ADDC ?A, B
? ? MOV ? 40H, A ?;(40H 41H)=千位*1000+百位*100+十位*10+個(gè)位
;---------------------
? ? MOV ? A, #10H ? ? ;2710H=1萬(wàn)
? ? MOV ? B, 30H ? ? ?;萬(wàn)位
? ? MUL ? AB
? ? MOV ? R2, B
? ? MOV ? R3, A
? ? MOV ? A, #27H
? ? MOV ? B, 30H ? ? ?;萬(wàn)位
? ? MUL ? AB
? ? ADD ? A, R2
? ? MOV ? R2, A ? ? ? ;R2 R3=萬(wàn)位*10000
;---------------------
? ? MOV ? A, R3
? ? ADD ? A, 41H
? ? MOV ? 41H, A
? ? MOV ? A, R2
? ? ADDC ?A, 40H
? ? MOV ? 40H, A
RET
本程序已經(jīng)通過(guò)了試驗(yàn)驗(yàn)證,保證正確。
后記:
針對(duì)這個(gè)問(wèn)題,提問(wèn)者采納了一個(gè)錯(cuò)誤的程序。
這個(gè)錯(cuò)誤程序,不僅有錯(cuò),長(zhǎng)度還差不多達(dá)到本程序的兩倍。
做而論道編寫(xiě)的程序,不僅正確,基本上就是最精簡(jiǎn)的,大家可以比較看一下。
其實(shí),做而論道還有更為精簡(jiǎn)的,僅僅才 35 行指令,可以說(shuō)是世上最精練轉(zhuǎn)換程序。只是技巧太高,不易理解,就不公布了。
用 C 語(yǔ)言編程,不太合乎題目的要求。
但是,也有人編寫(xiě)的很好,收錄在下面:
viod BcdHex(viod)
{
? ? unsigned short int *OutAdd, *SrcAdd, x;
??
? ? SrcAdd = 0x30;
? ? OutAdd = 0x40;
? ? *OutAdd = 0; ? ?x = 1;
? ? *OutAdd += (SrcAdd[0] & 0x0F) * x; x *= 10;?
? ? *OutAdd += (SrcAdd[0] ?>> ?4) * x; x *= 10;
? ? *OutAdd += (SrcAdd[1] & 0x0F) * x; x *= 10;?
? ? *OutAdd += (SrcAdd[1] ?>> ?4) * x; x *= 10;
? ? *OutAdd += (SrcAdd[2] & 0x0F) * x;
}