<meter id="pryje"><nav id="pryje"><delect id="pryje"></delect></nav></meter>
          <label id="pryje"></label>

          新聞中心

          EEPW首頁 > 嵌入式系統(tǒng) > 設計應用 > MCS-51單片機匯編指令詳解

          MCS-51單片機匯編指令詳解

          作者: 時間:2016-12-01 來源:網絡 收藏

                這有什么意義呢?ACC中的值本來就是100,B中的值本來就是20,是的,在本例中,的確沒有意義,但在實際工作中,則在PUSH B后往往要執(zhí)行其他指令,而且這些指令會把A中的值,B中的值改掉,所以在程序的結束,如果我們要把A和B中的值恢復原值,那么這些指令就有意義了。

                還有一個問題,如果我不用堆棧,比如說在PUSH ACC指令處用MOV 60H,A,在PUSH B處用指令MOV 61H,B,然后用MOV A,60H,MOV B,61H來替代兩條POP指令,不是也一樣嗎?是的,從結果上看是一樣的,但是從過程看是不一樣的,PUSH和POP指令都是單字節(jié),單周期指令,而 MOV指令則是雙字節(jié),雙周期指令。更何況,堆棧的作用不止于此,所以一般的計算機上都設有堆棧,而我們在編寫子程序,需要保存數(shù)據(jù)時,通常也不采用后面的方法,而是用堆棧的方法來實現(xiàn)。

                  例:寫出以下程序的運行結果

                    MOV 30H,#12

                    MOV 31H,#23

                    PUSH 30H

                    PUSH 31H

                    POP 30H

                    POP 31H

                    結果是30H中的值變?yōu)?3,而31H中的值則變?yōu)?2。也就兩者進行了數(shù)據(jù)交換。從這個例子可以看出:使用堆棧時,入棧的書寫順序和出棧的書寫順序必須相反,才能保證數(shù)據(jù)被送回原位,否則就要出錯了。

              算術運算類指令

              1.不帶進位位的加法指令

                ADD A,#DATA ;例:ADD A,#10H

                ADD A,direct ;例:ADD A,10H

                ADD A,Rn ;例:ADD A,R7

                ADD A,@Ri ;例:ADD A,@R0

                用途:將A中的值與其后面的值相加,最終結果否是回到A中。

               例:

                 MOV A,#30H

                ADD A,#10H

                則執(zhí)行完本條指令后,A中的值為40H。

              2.帶進位位的加法指令

                ADDC A,Rn

                ADDC A,direct

                ADDC A,@Ri

                ADDC A,#data

                用途:將A中的值和其后面的值相加,并且加上進位位C中的值。

                說明:由于51單片機是一種8位機,所以只能做8位的數(shù)學運算,但8位運算的范圍只有0-255,這在實際工作中是不夠的,因此就要進行擴展,一般是將2個8位的數(shù)學運算合起來,成為一個16位的運算,這樣,可以表達的數(shù)的范圍就可以達到0-65535。如何合并呢?其實很簡單,讓我們看一個 10進制數(shù)的例子:

                  66+78。

                這兩個數(shù)相加,我們根本不在意這的過程,但事實上我們是這樣做的:先做6+8(低位),然后再做6+7,這是高位。做了兩次加法,只是我們做的時候并沒有刻意分成兩次加法來做罷了,或者說我們并沒有意識到我們做了兩次加法。之所以要分成兩次來做,是因為這兩個數(shù)超過了一位數(shù)所能表達的范置(0-9)。

                在做低位時產生了進位,我們做的時候是在適當?shù)奈恢命c一下,然后在做高位加法是將這一點加進去。那么計算機中做16位加法時同樣如此,先做低 8位的,如果兩數(shù)相加產生了進位,也要“點一下”做個標記,這個標記就是進位位C,在PSW中。在進行高位加法是將這個C加進去。例:1067H+10A0H,先做67H+A0H=107H,而107H顯然超過了0FFH,因此最終保存在A中的是7,而1則到了PSW中的CY位了,換言之,CY就相當于是100H。然后再做10H+10H+CY,結果是21H,所以最終的結果是2107H。

              3.帶借位的減法指令

                SUBB A,Rn

                SUBB A,direct

                SUBB A,@Ri

                SUBB A,#data

                設(每個H,(R2)=55H,CY=1,執(zhí)行指令SUBB A,R2之后,A中的值為73H。

                說明:沒有不帶借位的減法指令,如果需要做不帶位的減法指令(在做第一次相減時),只要將CY清零即可。

              4.乘法指令

                MUL AB

                此指令的功能是將A和B中的兩個8位無符號數(shù)相乘,兩數(shù)相乘結果一般比較大,因此最終結果用1個16位數(shù)來表達,其中高8位放在B中,低8位放在A中。在乘積大于FFFFFH(65535)時,0V置1(溢出),否則OV為0,而CY總是0。

                例:(A)=4EH,(B)=5DH,執(zhí)行指令

                MUL AB后,乘積是1C56H,所以在B中放的是1CH,而A中放的則是56H。

              5.除法指令

                DIV AB

                此指令的功能是將A中的8位無符號數(shù)除以B中的8位無符號數(shù)(A/B)。除法一般會出現(xiàn)小數(shù),但計算機中可沒法直接表達小數(shù),它用的是我們小學生還沒接觸到小數(shù)時用的商和余數(shù)的概念,如13 /5,其商是2,余數(shù)是3。除了以后,商放在A中,余數(shù)放在B中。CY和OV都是0。如果在做除法前B中的值是00H,也就是除數(shù)為0,那么0V=1。

              6.加1指令

                INC A

                INC Rn

                INC direct

                INC @Ri

                INC DPTR

                用途很簡單,就是將后面目標中的值加1。例:(A)=12H,(R0)=33H,(21H)=32H,(34H)=22H,DPTR=1234H。執(zhí)行下面的指令:

                INC A (A)=13H

                INC R2 (R0)=34H

                INC 21H (21H)=33H

                INC @R0 (34H)=23H

                INC DPTR 9; ( DPTR)=1235H

                結果如上所示。

                說明:從結果上看INC A和ADD A,#1差不多,但INC A是單字節(jié),單周期指令,而ADD #1則是雙字節(jié),雙周期指令,而且INC A不會影響PSW位,如(A)=0FFH,INC A后(A)=00H,而CY依然保持不變。如果是ADD A ,#1,則(A)=00H,而CY一定是1。因此加1指令并不適合做加法,事實上它主要是用來做計數(shù)、地址增加等用途。另外,加法類指令都是以A為核心的其中一個數(shù)必須放在A中,而運算結果也必須放在A中,而加1類指令的對象則廣泛得多,可以是寄存器、內存地址、間址尋址的地址等等。

              7.減1指令

                DEC A

                DEC RN

                DEC direct

                DEC @Ri

                與加1指令類似,就不多說了。

              邏輯運算類指令:

              1.對累加器A的邏輯操作:

                CLR A ;將A中的值清0,單周期單字節(jié)指令,與MOV A,#00H效果相同。

                CPL A ;將A中的值按位取反

                RL A ;將A中的值邏輯左移

                RLC A ;將A中的值加上進位位進行邏輯左移

                RR A ;將A中的值進行邏輯右移

                RRC A ;將A中的值加上進位位進行邏輯右移

                SWAP A ;將A中的值高、低4位交換。

                例:(A)=73H,則執(zhí)行CPL A,這樣進行:

                73H化為二進制為01110011,

                逐位取反即為 10001100,也就是8CH。

                RL A是將(A)中的值的第7位送到第0位,第0位送1位,依次類推。

                例:A中的值為68H,執(zhí)行RL A。68H化為二進制為01101000,按上圖進行移動。01101000化為11010000,即D0H。

                RLC A,是將(A)中的值帶上進位位(C)進行移位。

                例:A中的值為68H,C中的值為1,則執(zhí)行RLC A

                1 01101000后,結果是0 11010001,也就是C進位位的值變成了0,而(A)則變成了D1H。

                RR A和RRC A就不多談了,請大家參考上面兩個例子自行練習吧。

                SWAP A,是將A中的值的高、低4位進行交換。

                例:(A)=39H,則執(zhí)行SWAP A之后,A中的值就是93H。怎么正好是這么前后交換呢?因為這是一個16進制數(shù),每1個16進位數(shù)字代表4個二進位。注意,如果是這樣的:(A)=39,后面沒H,執(zhí)行SWAP A之后,可不是(A)=93。要將它化成二進制再算:39化為二進制是10111,也就是0001,0111高4位是0001,低4位是0111,交換后是01110001,也就是71H,即113。

              2.邏輯與指令

                ANL A,Rn ;A與Rn中的值按位與,結果送入A中

                ANL A,direct ;A與direct中的值按位與,結果送入A中

                ANL A,@Ri ;A與間址尋址單元@Ri中的值按位與,結果送入A中

                ANL A,#data ;A與立即數(shù)data按位與,結果送入A中

                ANL direct,A ;direct中值與A中的值按位與,結果送入direct中

                ANL direct,#data ;direct中的值與立即數(shù)data按位與,結果送入direct中。

                這幾條指令的關鍵是知道什么是邏輯與。這里的邏輯與是指按位與

                例:71H和56H相與則將兩數(shù)寫成二進制形式:

               ?。?1H) 01110001

               ?。?6H) 00100110

                結果 00100000 即20H,從上面的式子可以看出,兩個參與運算的值只要其中有一個位上是0,則這位的結果就是0,兩個同是1,結果才是1。



          評論


          相關推薦

          技術專區(qū)

          關閉
          看屁屁www成人影院,亚洲人妻成人图片,亚洲精品成人午夜在线,日韩在线 欧美成人 (function(){ var bp = document.createElement('script'); var curProtocol = window.location.protocol.split(':')[0]; if (curProtocol === 'https') { bp.src = 'https://zz.bdstatic.com/linksubmit/push.js'; } else { bp.src = 'http://push.zhanzhang.baidu.com/push.js'; } var s = document.getElementsByTagName("script")[0]; s.parentNode.insertBefore(bp, s); })();