1) c忌諱絕對定位。 常看見初學者要求使用_at_,這是一種謬誤,把c當作asm看待了。在c中變量的定位是編譯器的事情,初學者只要定義變量和變量的作 用域,編譯器就把一個固定地址給這個變量。怎么取得這個變量的地址?要用指針。比如unsigned char data x;后,x的地址就是&x, 你只要查看這個參數(shù),就可以在程序中知道具體的地址了。所以俺一看見要使用絕對定位的人,第一印象就是:這大概是個初學者。
2) 設置sp的問題。 原因和1差不對,編譯器在把所有變量和緩沖區(qū)賦予地址后,自動把最后一個字節(jié)開始的地方,作為sp的開始位置,所以初學者是不必 要去理會的。這體現(xiàn)c的優(yōu)越性,很多事情c編譯時候做了。
3) 用c的主程序結構: #include void main(void) { while(1); } 這是個最小的成功的c程序,包括頭部文件和程序主體。 頭部文件的名詞解釋:引用的外部資源文件,這個文件包括了硬件信息和外部模塊提供的可使用的函數(shù)和變量的說明??梢杂梦谋痉?式打開reg52.h,仔細研究下,會有一些寫程序的體會。
4) 這樣構成一個c項目 在c中,常用項目來管理。項目一般分為兩大塊:c文件塊和頭部文件塊。 我們常把不同功能寫在不同的c文件中,依*項目的管理,最后把所有文件連接起來,這樣就可以得到可以燒錄的hex文件或bin文件。 這些c文件中,有且只有唯一一個包括main()函數(shù),和3)中一樣的c文件。 用頭部文件把各個不同的c互相連接起來。一個c文件基本上要對應有一個h頭部文件,這個h文件就包含本c文件中可以提供給外面使 用的變量和函數(shù),沒有在h文件中列出的募梢運閌歉肅文件的內(nèi)部函數(shù)和變量,外部c不能使用。 例子:a.c: unsigned char i; unsigned char mwork; void test1(void) { mwork++; } void test2(void) { i++; } a.h文件中: extern unsigned char i; extern void test1(void); 這樣主程序m.c中: #include /*c編譯器內(nèi)部自帶的h文件,使用<>*/ #i nclude 'a.h' /*自定義的h文件,一般用''*/ void main(void) { test1(); /*使用a.c模塊文件中的函數(shù)*/ while(1){ i++; /*使用a.c模塊文件中的變量*/ } }
5) 51家族 核心都是基于8031的,有很多在此核心上進行擴展,有的把程序存儲器放在內(nèi)部:89c(s)51..,有的增加了ram:89c(s)52..,有的增加 了一些專用硬件80c552...,有的改變時鐘時序w77e58...。市面上現(xiàn)在常用的主要有atmel公司的at89x系列,philips的p87(89)x,臺 灣winbond的w77(78)x系列,cygnal的c8051fx系列。
6) 51單片機結構的c描述 這里不講51的具體結構,只是引導初學者快速理解51單片機的物理結構。寄存器和io及其它硬件設備的地址名稱,在相應的c頭部文件 中可以找到。51為reg51.h,52為reg52.h,以次類推,比如winbond的78e58就為w78e58.h這些h文件中的描述: srf,定義一個8位的設備。 srf16,定義一個16位的設備。 sbit,定義一個位的設備。 用這些語句定義后,就可以在c中象匯編一樣使用這些硬件設備,這是單片機應用比標準c特殊的地方,其它差別很少。
7) 在51系列中data,idata,xdata,pdata的區(qū)別 data:固定指前面0x00-0x7f的128個ram,可以用acc直接讀寫的,速度最快,生成的代碼也最小。 idata:固定指前面0x00-0xff的256個ram,其中前128和data的128完全相同,只是因為訪問的方式不同。idata是用類似c中的指針方式 訪問的。匯編中的語句為:mox acc,@rx.(不重要的補充:c中idata做指針式的訪問效果很好) xdata:外部擴展ram,一般指外部0x0000-0xffff空間,用dptr訪問。 pdata:外部擴展ram的低256個字節(jié),地址出現(xiàn)在a0-a7的上時讀寫,用movx acc,@rx讀寫。這個比較特殊,而且c51好象有對此bug, 建議少用。但也有他的優(yōu)點,具體用法屬于中級問題,這里不提。
8) startup.a51的作用 和匯編一樣,在c中定義的那些變量和數(shù)組的初始化就在startup.a51中進行,如果你在定義全局變量時帶有數(shù)值,如unsigned char data xxx=100;,那startup.a51中就會有相關的賦值。如果沒有=100,startup.a51就會把他清0。(startup.a51==變量的初始化)。 這些初始化完畢后,還會設置sp指針。對非變量區(qū)域,如堆棧區(qū),將不會有賦值或清零動作。 有人喜歡改startup.a51,為了滿足自己一些想當然的愛好,這是不必要的,有可能錯誤的。比如掉電保護的時候想保存一些變量, 但改startup.a51來實現(xiàn)是很笨的方法,實際只要利用非變量區(qū)域的特性,定義一個指針變量指向堆棧低部:0xff處就可實現(xiàn)。 為什么還要去改? 可以這么說:任何時候都可以不需要改startup.a51,如果你明白它的特性。