最近看到幾個尋求單片機入門的帖子,一時心血來潮,把自己的一些入門心得寫了下來,希望能對初學者有所幫助吧。
可能很多人學習單片機的開始都是一章一章的的去閱讀教程,我也這樣做過,結果就是沒多久就昏昏欲睡了。對于初學者來說,什么隨機存儲器啊,只讀存儲器啊,寄存器啊,尋址方式啊,周期啊,指令啊。。。等等等等,簡直就跟看天書一樣。其實,我認為對于初學者來說,沒必要了解這么多,學習總是一個循序漸進的過程,不要妄想著能一下子就把單片機的理解透了,然后再去動手做實驗,做項目,這是很不現實的。
學習單片機的時候,要想著單片機能做什么我就學什么,我想要做什么就學什么,不懂,就翻書,再不行,就上網找。那么首先單片機能做些什么呢?單片機能做的事情很多很多,恐怕說個幾天幾夜都說不完。可能很多人會這么說,這么多的功能,這么多的例子,究竟從何學起啊!但是在我看來,單片機能做的只有兩件事而你要做的也只有這兩件事情:第一,輸出高低電平;第二,接收高低電平的輸入。假如單片機沒有輸入輸出功能,那么程序編得在怎么超凡脫俗,也沒有任何意義。因為,沒有了跟外圍器件的通信,單片機還有什么用呢!那么跟外圍器件的通信靠的是什么呢?高電平(+3.3V或+5V)和低電平(0V)。那么我們的目的就很明確了,學習單片機的目的就是讓單片機的各個管腳輸入或輸出高或低電平。在程序上代表高低電平的就是數字量1和0。也就是說,程序的最終目的就是在各個管腳上輸入或輸出1或0。所有的程序都是為了達成這個目的而設計的。換句話來說,只要能在你想要的管腳輸入或輸出你想要實現的高或低電平,那么你的目的就已經達到了,不要去管你的程序有多么的臃腫或是不堪入目,這個會隨著你學習的深入和經驗的積累而逐漸改善,不需要著急。
舉個最簡單的例子,在單片機的P1.0的管腳上接一個LED燈,要讓LED燈點亮,就是在P1.0管腳上輸出高電平,要讓LED燈熄滅,就是在P1.0腳上輸出低電平。那么怎么樣才能在P1.0腳上輸出高或低電平呢?不知道,那就去翻書一條一條的去找指令。哦,找到一條SETB置位指令,置位P1.0那不就是把1賦給P1.0嗎,P1.0置1,不就是輸出高電平了嗎?至于是不是,誰試誰知道。不過,先不要著急,既然找到了輸出高電平的指令,那么順便找找輸出低電平的指令。好了,沒錯,就是你了CLR。那么現在就可以編程序了:
ORG 0000H
JMP MAIN
OGR 0030H ;如果不能理解這幾條指令的意思,那就直接套用就可以了
MAIN:
SETB P1.0 ;輸出高電平,點亮LED燈
CLR P1.0 ;輸出低電平,熄滅LED燈
END
好了,程序完成,很簡單吧。可是,這個只是一亮一滅,我要它不停的閃爍怎么辦?簡單!多加一句跳轉指令就行了,跳轉指令上面就有JMP,那好吧,再改一下程序
ORG 0000H
JMP MAIN
OGR 0030H
MAIN:
SETB P1.0
CLR P1.0
JMP MAIN
END
大功告成
可是,程序運行之后,看不到LED燈一亮一滅啊!怎么回事?這是當然了,單片機CUP的運行速度是以微秒來計的,人的眼睛是反應不過來的。那要怎么辦呢?讓CPU停一下等個一兩秒再執行下一條指令?那顯然不行,地球人都知道。那就找點事情給CPU去忙吧,不管它干什么都行,只要再這段時間內不要去碰P1.0管腳就行了。那么讓它去做什么呢,國際上-_-!!!通常讓它去數數,因為CPU每數一個數的時間都是一樣的,比如說1微秒,那么數1 000個數,就是1毫秒,數1 000 000個數就是一秒。那么怎么樣讓CPU去數數呢?繼續找指令表,我找。。。找到一個INC,每執行一次,操作數加1,那我要數到1 000 000的時候停止呢,怎么辦?不知道。不知道!那要你干什么,一邊去吧你,順便把你兄弟DEC也帶走,我不想再見到你們!我再找。。。這個好像有點用JZ,累加器A中為0就跳轉,好像可以啊,我先讓CPU跳一邊去然后給A一個數1 000 000,讓A從1 000 000減到0,A為0時再跳轉回來不就行了?不過累加器A是什么?不知道?那就再翻書。。。哦,好像A最大只能到255,到不了1 000 000,怎么辦?255就255吧,先試試再說,看能不能看出變化。那么怎么給A送數呢?MOV唄!好了,那誰誰誰,你給我回來,DEC別看了,說的就是你!嗯,再改一下程序
ORG 0000H
JMP MAIN
OGR 0030H
MAIN:
SETB P1.0
MOV A,#255 ;給A一個數,讓CPU去數
JMP WAIT ;CPU給我一邊數數去
LED_OFF:
CLR P1.0
MOV A,#255 ;
JMP WAIT1 ;再來一個
LED_ON:
JMP MAIN
WAIT:
DEC A ;A-1
JZ LED_OFF ;等于0就跳回去
JMP WAIT ;不等于0就繼續減
WAIT1:
DEC A ;A-1
JZ LED_ON ;等于0就跳回去
JMP WAIT1 ;不等于0就繼續減
END
編譯,,排錯,運行,大功告成 
好了,程序編完了,也能運行了,不過現在高興是不是太早了,你在JMP來JMP去的,JMP的我頭都暈了,那我要是要再延長一點時間,你豈不是要JMP個沒完沒了了?!難道就沒有別的方法了嗎?那好吧,我在翻翻書。真是書到用時方恨少啊。。。咦,這個看起來有點意思,CALL,是不是跟打電話一樣,不管你在哪里,一個CALL,就能找到你啊。不過這個ACALL和LCALL又有神馬不同呢,難道還有國內長途和國際長途之分?不管了,就用你了LCALL,反正不用花錢。
ORG 0000H
JMP MAIN
OGR 0030H
MAIN:
SETB P1.0
MOV A,#255
LCALL WAIT ;我CALL
LCALL WAIT ;我再CALL
LCALL WAIT ;
LCALL WAIT ;
LCALL WAIT ;我CALL,CALL,CALL。。。
CLR P1.0
LCALL WAIT
LCALL WAIT
LCALL WAIT
LCALL WAIT ;哈哈哈。。。CALL個夠,爽
JMP MAIN
WAIT:
DEC A ;A-1
JNZ WAIT ;沒數完,繼續。。。
RET ;數完了,那我掛電話了,有時間再CALL你啊
END
好了,這回看起來舒服多了。不過累加器A,看起來你有點意見?A:“廢話!你不知道老子很忙的嗎!分分鐘幾十萬上下,你叫我給你數數?你確定,你的腦袋沒被驢給踢過?老子縱橫機湖幾十年,閱人無數,就沒見過你這么白的程序員!”好吧,大哥,你牛,我惹不起你我躲的起。我再翻書,幸好這不是在考試,我想怎么翻就怎么翻。。。有了!就是你了DJNZ,減1不為0就跳轉。咦,怎么沒有減1為0跳轉的呢?也不知道創造匯編的那位大神是怎么想的。好吧,這不是我們這些小菜鳥該管的,還是改我的程序比較靠譜一點
ORG 0000H
JMP MAIN
OGR 0030H
MAIN:
SETB P1.0
MOV R0,#255 ;那就換一個唄
LCALL WAIT ;我CALL
LCALL WAIT ;我再CALL
LCALL WAIT ;
LCALL WAIT ;
LCALL WAIT ;我CALL,CALL,CALL。。。
CLR P1.0
LCALL WAIT
LCALL WAIT
LCALL WAIT
LCALL WAIT ;哈哈哈。。。
JMP MAIN
WAIT:
DJNZ R0,WAIT ;沒數完,繼續
RET ;數完了,那我掛電話了,有時間再CALL你啊
END
好了,終于終于終于編完了,其實單片機也不怎么難嘛,呵呵。
最后,再介紹一句,其實
DJNZ R0,WAIT
這句,還可以換成
DJNZ R0,$
這樣,減1不為0就等待,其實我想介紹的是這一句
JMP $
這是個死循環,原地跳步,用來調試程序是非常好用的。不知道創造這句的大神是不是要告訴全世界的程序員,美元的魅力連CPU也擋不住,看到它,誰也跑不動。好了,言歸正傳,這一句其實用來調試程序是非常好用的,不知道怎么用,就先記住吧,或許以后有用,或許永遠也沒用,一家之言,每個人有每個人的方法。
好了,就講到這里了,一時興起講了這么多,以后可能再也沒有這么好的興致寫這么多東西了,就這樣吧
|