1.關(guān)于并行I/O口的“讀引腳”和“讀鎖存器”指令的區(qū)別
讀并行I/O口有“讀引腳”和“讀鎖存器”之分。以Pl口為例,當(dāng)P1口的P1.0引腳外接一個發(fā)光二極管LED的陽極,LED的陰極接地。若想查看一下單片機(jī)剛才向P1.0腳輸出的信息是0還是1,這時不能直接從P1.0讀取,因為單片機(jī)剛才向P1.0輸出的信息如果是1的話,則LED導(dǎo)通點(diǎn)亮,此時P1.0引腳就為0電平,如果直接讀引腳,結(jié)果顯然是錯誤的。正確的做法是讀D鎖存器的Q端狀態(tài),那里儲存的才是前一時刻送給Pl.0的真實值。也就是說,凡遇“讀取P1口前一狀態(tài)以便修改后再送出”的情形,都應(yīng)當(dāng)“讀鎖存器”的Q端信息,而不是讀取引腳的信息。
當(dāng)P1口外接輸入設(shè)備時,要想P1口引腳上反映的是真實的輸入信號,必須要設(shè)法先讓該引腳內(nèi)部的場效應(yīng)管截止才行(如圖2-7所示),否則當(dāng)場效應(yīng)管導(dǎo)通時,P1口引腳上將永遠(yuǎn)為低電平,無法正確反映外部設(shè)備的輸入信號。讓場效應(yīng)管截止,就是用指令給Pl口的相應(yīng)位送一個1電平,這就是為什么讀引腳之前,一定要先送出1的原因。
指令“MOV C,Pl.0”讀的是Pl.0引腳,同樣,指令“MOV A,Pl”也是讀引腳指令,讀引腳指令之前一定要有向Pl.0寫1的指令。而指令“CPL Pl.O”則是“讀鎖存器”,也即“讀-修改-寫”指令,它會先讀Pl.O的鎖存器的Q端狀態(tài),接著取反,然后再送到Pl.0引腳上。而指令“ANLPl,A”也是“讀鎖存器”命令。類似的“讀-修改-寫”指令舉例如下:
2.關(guān)于操作數(shù)的字節(jié)地址和位地址的區(qū)分問題
如何區(qū)別指令中出現(xiàn)的字節(jié)變量和位變量呢?例如指令“MOV C,40H”和指令“MOV A,40H”兩條指令中的助記符相同,但是指令中源操作數(shù)“40H”都是以直接地址形式給出的,“40H”究竟是字節(jié)地址還是位地址?對于助記符相同的指令,觀察操作數(shù)就可看出。顯然前條指令中的“40H”是位地址,因為目的操作數(shù)C是位變量。而后條指令中的“40H”是字節(jié)地址,因為目的操作數(shù)A是字節(jié)變量。當(dāng)然,對于助記符不同的指令,從助記符的形式,就可以看出其中指令究竟是“字節(jié)”操作,還是“位”操作。
3.關(guān)于累加器A與ACC的書寫問題
累加器可寫成A,也可寫成Acc,它們的區(qū)別是什么?在51單片機(jī)匯編語言指令中是有區(qū)別的。Acc在匯編后的機(jī)器碼必有一個字節(jié)的操作數(shù)是累加器的字節(jié)地址EOH,A在匯編后則隱含在指令操作碼中。例如指令“INC A”的機(jī)器碼,查表3-2是04H。如寫成“INC Acc”后,則成了“INC direct”的格式,再查表3-2,對應(yīng)的機(jī)器碼為“05H EOH”。在對累加器A的直接尋址和累加器A的某一位尋址要用Acc,而不能寫成A。例如,指令“POP Acc”不能寫成“POP A”;指令“SETB Acc.O”,不能寫成“SETB A.O”。
4.書寫2位十六進(jìn)制數(shù)據(jù)前要加“0”
在書寫源程序時經(jīng)常遇到必須在某些數(shù)據(jù)或地址的前面多填一個“前導(dǎo)0”的問題,否則在匯編成時機(jī)器語言匯編就通不過。這是匯編語言的嚴(yán)格性和規(guī)范性的具體體現(xiàn)。由于部分十六進(jìn)制數(shù)是用字母來表示的,而程序內(nèi)的標(biāo)號也常用字母表示,為了將標(biāo)號和數(shù)據(jù)區(qū)分開,幾乎所有的匯編語言都規(guī)定,凡是以字母開頭(對十六進(jìn)制數(shù)而言,就是A~F開頭)的數(shù)字量,應(yīng)當(dāng)在前面添加一個數(shù)字“0”。至于地址量,它也是數(shù)據(jù)量的一種,前面也應(yīng)該添加“0”。例如:
如果不加“前導(dǎo)0”,就會把字母開頭的數(shù)據(jù)量當(dāng)作標(biāo)號來處理,從而出錯以及不能通過匯編。