堆棧操作類(lèi)指令(2條)舉例——mcs51單片機(jī)匯編語(yǔ)言
掃描二維碼
隨時(shí)隨地手機(jī)看文章
堆棧操作類(lèi)指令(2條)
這4類(lèi)指令的作用是把直接尋址單元的內(nèi)容傳送到堆棧指針SP所指的單元中,以及把SP所指單元的內(nèi)容送到直接尋址單元中。這類(lèi)指令只有兩條,下述的第一條常稱(chēng)為入棧操作指令,第二條稱(chēng)為出棧操作指令。需要指出的是,單片機(jī)開(kāi)機(jī)復(fù)位后,(SP)默認(rèn)為07H,但一般都需要重新賦值,設(shè)置新的SP首址。入棧的第一個(gè)數(shù)據(jù)必須存放于SP+1所指存儲(chǔ)單元,故實(shí)際的堆棧底為SP+1所指的存儲(chǔ)單元。
PUSH data;(SP)+1→(SP),(data)→(SP) 堆棧指針首先加1,直接尋址單元中的數(shù)據(jù)送到堆棧指針SP所指的單元中
POP data;(SP)→(data)(SP)-1→(SP), 堆棧指針SP所指的單元數(shù)據(jù)送到直接尋址單元中,堆棧指針SP再進(jìn)行減1操作
第一條指令稱(chēng)為推入,就是將direct中的內(nèi)容送入堆棧中,第二條指令稱(chēng)為彈出,就是將堆棧中的內(nèi)容送回到direct中,推入指令的執(zhí)行過(guò)程是:首先將P中的值加1,然后把SP中的值當(dāng)做地址,將direct中的值送入以SP中的值為地址的RAM單元中去。
例:
MOV SP,#5FH
MOV A,#100
MOV B,#20
PUSH ACC
PUSH B
執(zhí)行第一條PUSH ACC指令是這樣的:將SP中的值加1,即變?yōu)?0H,然后將A中的值送到60H單元中,因此執(zhí)行完本條指令后,內(nèi)存60H單元中的值就是100,同樣執(zhí)行PUSH B時(shí),是將SP+1,即變?yōu)?1H,然后將B中的值送入到61H單元中,即執(zhí)行完本條指令后61H單元中的值變?yōu)?0。
POP指令的執(zhí)行也是這樣的,首先將SP中的值做為地址,并將此地址中的數(shù)送到POP指令的那個(gè)direct中,然后SP減1。
接上例:
POP B
POP ACC
則執(zhí)行過(guò)程是:將SP中的值(現(xiàn)在是61H)作為地址,取61H單元中的數(shù)值(現(xiàn)在是20),送到B中,所以執(zhí)行完本條指令后B中的值是20,然后SP減1,因此本條指令執(zhí)行完后,SP的值變?yōu)?0H,然后搪行POP ACC,將SP中的值(60H)作為地址,從該地址中取數(shù)(現(xiàn)在是100),并送到ACC中,所以執(zhí)行完本條指令后,ACC中的值是100。
這有什么意義呢?ACC中的值本來(lái)就是100,B中的值本來(lái)就是20,是的,在本例中,的確沒(méi)有意義,但在實(shí)際工作中,則在PUSH B后往事要執(zhí)行其它的指令,而且這些指令會(huì)把A中的值,B中的值改掉,所以在程序結(jié)束,如果我們要把A和B中的值恢復(fù)原值,那么這些指令就沒(méi)有意義了。
還有一個(gè)問(wèn)題,如果我們不用堆棧,比如說(shuō)在PUSH ACC指令處用MOV 60H,A在PUSH B處用指令MOV 61H,B,然后用MOV A,60H,MOV B,61H來(lái)替代兩處POP指令,不也是一樣嗎?是的,從結(jié)果上看是一樣的,但從過(guò)程看是不一樣的,PUSH和POP指令都是單字節(jié),單周期指令,而MOV指令則是雙字節(jié),雙周期指令,更何況,堆棧的作用不止于此,所以一般的計(jì)算機(jī)上都設(shè)有堆棧,而我們?cè)诰帉?xiě)子程序,需要保存數(shù)據(jù)時(shí),通常不采用后面的方法,而是用堆棧的方法來(lái)實(shí)現(xiàn)。
例:寫(xiě)出以下程序的運(yùn)行結(jié)果
MOV 30H,#12
MOV 31H,#23
PUSH 30H
PUSH 31H
POP 30H
POP 31H
結(jié)果是30H中的值變?yōu)?3,而31H中的值則變?yōu)?2。也就是兩者進(jìn)行了數(shù)據(jù)交換。從這個(gè)例子可以看出:使用堆棧時(shí),入棧的書(shū)寫(xiě)順序和出棧的書(shū)寫(xiě)順序必須相反,才能保證數(shù)據(jù)被送回原位,否則就出錯(cuò)了。
另外特別注意事項(xiàng):
進(jìn)行堆棧操作時(shí),我們不能:
PUSH R0
PUSH R1
而只能:
PUSH 00H
PUSH 01H
POP也是一樣