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個字節,地址出現在A0-A7的上時讀寫,用movx ACC,@Rx讀寫。這個比較特殊,而且C51好象有對此BUG, 建議少用。但也有他的優點,具體用法屬于中級問題,這里不提。 startup.a51 的作用,和匯編一樣,在C中定義的那些變量和數組的初始化就在startup.a51中進行,如果你在定義全局變量時帶有數值,如unsigned char dATa xxx="100";,那startup.a51中就會有相關的賦值。如果沒有=100,startup.a51就會把他清 0。(startup.a51==變量的初始化)。 這些初始化完畢后,還會設置SP指針。對非變量區域,如堆棧區,將不會有賦值或清零動作。 有 人喜歡改startup.a51,為了滿足自己一些想當然的愛好,這是不必要的,有可能錯誤的。比如掉電保護的時候想保存一些變量, 但改startup.a51來實現是很笨的方法,實際只要利用非變量區域的特性,定義一個指針變量指向堆棧低部:0xff處就可實現。, 為什么還要去改? 可以這么說:任何時候都可以不需要改startup.a51,如果你明白它的特性。 bit 是在內部數據存儲空間中 20H .. 2FH 區域中一個位的地址,這在DATA的20H以后以字節形式出現,可互相參照。另外加上8051 可尋址 的SFR,但剛剛試過,只是00H--7FH起作用,也就是說當數據有變化時顏色變紅,以后的從80H到--FFH就不是位尋址區了,是位尋址的特殊寄存器,如涉及到了可位尋址的那11個當然會有反應。 復位后,程序計數器PC的內容為0000H,內部RAM各單元的值不確定。各功能寄存器的復位值如下:堆棧指針SP的復位值為07H,累加器ACC、寄存器B的復位值為00H,數據指針DPTR的復位值為0000H,而p0、p1、p2、p3四個口的復位值為0FFH。其他SFR如PSW、TCON、TMOD、TL0、TH0、TL1、TH1的復位值也為00H。 wave中是低128字節和高128字節(0-7FH),低128字節是片內RAM區,高128字節(80-FFH)是SFR(特殊功能寄存器)bit則是位于低128字節的20H .. 2FH 區域,即data的20H .. 2FH 區域 code 是在 0000H .. 0FFFFH 之間的一個代碼地址。 我用 ORG 5000H TAB: DB 22H,3BH,43H,66H,5H,6DH,88H后, CODE從5000H開始以后變成DB各位 data 是在 0 到 127 之間的一個數據存儲器地址,或者加 128 .. 255 范圍內的一個特殊功能寄存器(SFR)地址。兩者訪問的方式不同。實際上由于PSW的復位設置PSW.3=RS0和PSW.4=RS1皆為0,所以通用工作寄存器區就是第0區,所以data的00--07H部分是與REG欄中的R0--R7對應的。以后的則僅代表低128字節的內部RAM。 idata 是 0 to 255 范圍內的一個 idata 存儲器地址。 idata與data重合低128字節,有的地方只有DATA表示256字節的片內RAM, xdata 是 0 to 65535 范圍內的一個 xdata 存儲器地址。 指針類型和存儲區的關系詳解 一、存儲類型與存儲區關系 data ---> 可尋址片內ram bdata ---> 可位尋址的片內ram idata ---> 可尋址片內ram,允許訪問全部內部ram pdata ---> 分頁尋址片外ram (MOVX @R0) (256 BYTE/頁) xdata ---> 可尋址片外ram (64k 地址范圍FFFFH) code ---> 程序存儲區 (64k 地址范圍),對應MOVC @DPTR 二、指針類型和存儲區的關系 對變量進行聲明時可以指定變量的存儲類型如: uchar data x和data uchar x相等價都是在內ram區分配一個字節的變量。 同樣對于指針變量的聲明,因涉及到指針變量本身的存儲位置和指針所指向的存儲區位置不同而進行相應的存儲區類型關鍵字的 使用如: uchar xdata * data pstr 是指在內ram區分配一個指針變量("*"號后的data關鍵字的作用),而且這個指針本身指向xdata區("*"前xdata關鍵字的作用), 可能初學C51時有點不好懂也不好記。沒關系,我們馬上就可以看到對應“*”前后不同的關鍵字的使用在編譯時出現什么情況。 ...... uchar xdata tmp[10]; //在外ram區開辟10個字節的內存空間,地址是外ram的0x0000-0x0009 ...... 第1種情況: uchar data * data pstr; pstr="tmp"; 首先要提醒大家這樣的代碼是有bug的, 他不能通過這種方式正確的訪問到tmp空間。 為什么?我們把編譯后看到下面的匯編 代碼: MOV 0x08,#tmp(0x00) ;0x08是指針pstr的存儲地址 看到了嗎!本來訪問外ram需要2 byte來尋址64k空間,但因為使用data關鍵字(在"*"號前的那個),所以按KeilC編譯環境來說 就把他編譯成指向內ram的指針變量了,這也是初學C51的朋友們不理解各個存儲類型的關鍵字定義而造成的bug。特別是當工程中的 默認的存儲區類為large時,又把tmp[10] 聲明為uchar tmp[10] 時,這樣的bug是很隱秘的不容易被發現。 第2種情況: uchar xdata * data pstr; pstr = tmp; 這種情況是沒問題的,這樣的使用方法是指在內ram分配一個指針變量("*"號后的data關鍵字的作用),而且這個指針本身指向 xdata區("*"前xdata關鍵字的作用)。編譯后的匯編代碼如下。 MOV 0x08,#tmp(0x00) ;0x08和0x09是在內ram區分配的pstr指針變量地址空間 MOV 0x09,#tmp(0x00) 這種情況應該是在這里所有介紹各種情況中效率最高的訪問外ram的方法了,請大家記住他。 第3種情況: uchar xdata * xdata pstr; pstr="tmp"; 這中情況也是對的,但效率不如第2種情況。編譯后的匯編代碼如下。 MOV DPTR, #0x000A ;0x000A,0x000B是在外ram區分配的pstr指針變量地址空間 MOV A, #tmp(0x00) MOV @DPTR, A INC DPTR MOV A, #tmp(0x00) MOVX @DPTR, A 這種方式一般用在內ram資源相對緊張而且對效率要求不高的項目中。 第4種情況: uchar data * xdata pstr; pstr="tmp"; 如果詳細看了第1種情況的讀者發現這種寫法和第1種很相似,是的,同第1 種情況一樣這樣也是有bug的,但是這次是把pstr分 配到了外ram區了。編譯后的匯編代碼如下。 MOV DPTR, #0x000A ;0x000A是在外ram區分配的pstr指針變量的地址空間 MOV A, #tmp(0x00) MOVX @DPTR, A 第5種情況: uchar * data pstr; pstr="tmp"; 大家注意到"*"前的關鍵字聲明沒有了,是的這樣會發生什么事呢?下面這么寫呢!對了用齊豫的一首老歌名來說就是 “請跟我 來”,請跟我來看看編譯后的匯編代碼,有人問這不是在講C51嗎? 為什么還要給我們看匯編代碼。C51要想用好就要盡可能提升C51 編譯后的效率,看看編譯后的匯編會幫助大家盡快成為生產高效C51代碼的高手的。還是看代碼吧! MOV 0x08, #0X01 ;0x08-0x0A是在內ram區分配的pstr指針變量的地址空間 MOV 0x09, #tmp(0x00) MOV 0x0A, #tmp(0x00) 注意:這是新介紹給大家的,大家會疑問為什么在前面的幾種情況的pstr指針變量都用2 byte空間而到這里就用3 byte空間了 呢?這是KeilC的一個系統內部處理,在KeilC中一個指針變量最多占用 3 byte空間,對于沒有聲明指針指向存儲空間類型的指針, 系統編譯代碼時都強制加載一個字節的指針類型分辯值。具體的對應關系可以參考KeilC的help中C51 User's Guide。 第6種情況: uchar * pstr; pstr="tmp"; 這是最直接最簡單的指針變量聲明,但他的效率也最低。還是那句話,大家一起說好嗎!編譯后的匯編代碼如下。 MOV DPTR, #0x000A ;0x000A-0x000C是在外ram區分配的pstr指針變量地址空間 MOV A, #0x01 MOV @DPTR, A INC DPTR MOV DPTR, #0x000A MOV A, #tmp(0x00) MOV @DPTR, A INC DPTR MOV A, #tmp(0x00) MOVX @DPTR, A 這種情況很類似第5種和第3種情況的組合,既把pstr分配在外ram空間了又增加了指針類型的分辨值。 |
電解電容在電路中的作用 |
1,濾波作用,在電源電路中,整流電路將交流變成脈動的直流,而在整流電路之后接入一個較大容量的電解電容,利用其充放電特性,使整流后的脈動直流電壓變成相對比較穩定的直流電壓。在實際中,為了防止電路各部分供電電壓因負載變化而產生變化,所以在電源的輸出端及負載的電源輸入端一般接有數十至數百微法的電解電容.由于大容量的電解電容一般具有一定的電感,對高頻及脈沖干擾信號不能有效地濾除,故在其兩端并聯了一只容量為0.001--0.lpF的電容,以濾除高頻及脈沖干擾. 2,耦合作用:在低頻信號的傳遞與放大過程中,為防止前后兩級電路的靜態工作點相互影響,常采用電容藕合.為了防止信號中韻低頻分量損失過大,一般總采用容量較大的電解電容。 二、電解電容的判斷方法 電解電容常見的故障有,容量減少,容量消失、擊穿短路及漏電,其中容量變化是因電解電容在使用或放置過程中其內部的電解液逐漸干涸引起,而擊穿與漏電一般為所加的電壓過高或本身質量不佳引起。判斷電源電容的好壞一般采用萬用表的電阻檔進行測量.具體方法為:將電容兩管腳短路進行放電,用萬用表的黑表筆接電解電容的正極。紅表筆接負極(對指針式萬用表,用數字式萬用表測量時表筆互調),正常時表 針應先向電阻小的方向擺動,然后逐漸返回直至無窮大處。表針的擺動幅度越大或返回的速度越慢,說明電容的容量越大,反之則說明電容的容量越小.如表針指在中間某處不再變化,說明此電容漏電,如電阻指示值很小或為零,則表明此電容已擊穿短路.因萬用表使用的電池電壓一般很低,所以在測量低耐壓的電容時比較準確,而當電容的耐壓較高時,打時盡管測量正常,但加上高壓時則有可能發生漏電或擊穿現象. 三、電解電容的使用注意事項 1、電解電容由于有正負極性,因此在電路中使用時不能顛倒聯接。在電源電路中,輸出正電壓時電解電容的正極接電源輸出端,負極接地,輸出負電壓時則負極接輸出端,正極接地.當電源電路中的濾波電容極性接反時,因電容的濾波作用大大降低,一方面引起電源輸出電壓波動,另一方面又因反向通電使此時相當于一個電阻的電解電容發熱.當反向電壓超過某值時,電容的反向漏電電阻將變得很小,這樣通電工作不久,即可使電容因過熱而炸裂損壞. 2.加在電解電容兩端的電壓不能超過其允許工作電壓,在設計實際電路時應根據具體情況留有一定的余量,在設計穩壓電源的濾波電容時,如果交流電源電壓為220~時變壓器次級的整流電壓可達22V,此時選擇耐壓為25V的電解電容一般可以滿足要求.但是,假如交流電源電壓波動很大且有可能上升到250V以上時,最好選擇耐壓30V以上的電解電容。 3,電解電容在電路中不應靠近大功率發熱元件,以防因受熱而使電解液加速干涸. 4、對于有正負極性的信號的濾波,可采取兩個電解電容同極性串聯的方法,當作一個無極性的電容。 |
電容器在電子電路中幾乎是不可缺少的儲能元件,它具有隔斷直流、連通交流、阻止低頻的特性。廣泛應用在耦合、隔直、旁路、濾波、調諧、能量轉換和自動控制等電路中。熟悉電容器在不同電路中的名稱意義,有助于我們讀懂電子電路圖。
1.濾波電容 它接在直流電源的正、負極之間,以濾除直流電源中不需要的交流成分,使直流電平滑。一般常采用大容量的電解電容器,也可以在電路中同時并接其他類型的小容量電容以濾除高頻交流電。
2.退耦電容 并接于放大電路的電源正、負極之間,防止由電源內阻形成的正反饋而引起的寄生振蕩。
3.旁路電容 在交、直流信號的電路中,將電容并接在電阻兩端或由電路的某點跨接到公共電位上,為交流信號或脈沖信號設置一條通路,避免交流信號成分因通過電阻產生壓降衰減。
4.耦合電容 在交流信號處理電路中,用于連接信號源和信號處理電路或者作兩放大器的級間連接,用以隔斷直流,讓交流信號或脈沖信號通過,使前后級放大電路的直流工作點互不影響。
5.調諧電容 連接在諧振電路的振蕩線圈兩端,起到選擇振蕩頻率的作用。
6.襯墊電容 與諧振電路主電容串聯的輔助性電容,調整它可使振蕩信號頻率范圍變小,并能顯著地提高低頻端的振蕩頻率。適當地選定襯墊電容的容量,可以將低端頻率曲線向上提升,接近于理想頻率跟蹤曲線。
7.補償電容 它是與諧振電路主電容并聯的輔助性電容,調整該電容能使振蕩信號頻率范圍擴大。
8.中和電容 并接在三極管放大器的基極與發射極之間,構成負反饋網絡,以抑制三極管極間電容造成的自激振蕩。
9.穩頻電容 在振蕩電路中,起穩定振蕩頻率的作用。
10.定時電容 在RC時間常數電路中與電阻R串聯,共同決定充放電時間長短的電容。
11.加速電容 接在振蕩器反饋電路中,使正反饋過程加速,提高振蕩信號的幅度。
12.縮短電容 在UHF高頻頭電路中,為了縮短振蕩電感器長度而串接的電容。
13.克拉潑電容 在電容三點式振蕩電路中,與電感振蕩線圈串聯的電容,起到消除晶體管結電容對頻率穩定性影響的作用。
14.錫拉電容 在電容三點式振蕩電路中,與電感振蕩線圈兩端并聯的電容,起到消除晶體管結電容的影響,使振蕩器在高頻端容易起振。
15.穩幅電容 在鑒頻器中,用于穩定輸出信號的幅度。
16.預加重電容 為了避免音頻調制信號在處理過程中造成對分頻量衰減和丟失,而設置的RC高頻分量提升網絡電容。
17.去加重電容 為恢復原伴音信號,要求對音頻信號中經預加重所提升的高頻分量和噪聲一起衰減掉,設置在RC網絡中的電容。
18.移相電容 用于改變交流信號相位的電容。
19.反饋電容 跨接于放大器的輸入與輸出端之間,使輸出信號回輸到輸入端的電容。
20.降壓限流電容 串聯在交流電回路中,利用電容對交流電的容抗特性,對交流電進行限流,從而構成分壓電路。
21.逆程電容 用于行掃描輸出電路,并接在行輸出管的集電極與發射極之間,以產生高壓行掃描鋸齒波逆程脈沖,其耐壓一般在1500V以上。
22.校正電容 串接在偏轉線圈回路中,用于校正顯像管邊緣的延伸線性失真。
23.自舉升壓電容 利用電容器的充、放電儲能特性提升電路某點的電位,使該點電位達到供電端電壓值的2倍。
24.消亮點電容 設置在視放電路中,用于關機時消除顯像管上殘余亮點的電容。
25.軟啟動電容 一般接在開關電源的開關管基極上,防止在開啟電源時,過大的浪涌電流或過高的峰值電壓加到開關管基極上,導致開關管損壞。
26.啟動電容 串接在單相電動機的副繞組上,為電動機提供啟動移相交流電壓。在電動機正常運轉后與副繞組斷開。
27.運轉電容 與單相電動機的副繞組串聯,為電動機副繞組提供移相交流電流。在電動機正常運行時,與副繞組保持串接。
指針就是指變量或數據所在的存儲區地址。如一個字符型的變量 STR 存放在內存單元DATA 區的 51H 這個地址中,那么 DATA 區的 51H 地址就是變量 STR 的指針。在 C 語言中 指針是一個很重要的概念,正確有效的使用指針類型的數據,能更有效的表達復雜的數據 結構,能更有效的使用數組或變量,能方便直接的處理內存或其它存儲區。指針之所以 能這么有效的操作數據,是因為無論程序的指令、常量、變量或特殊寄存器都要存放在內 存單元或相應的存儲區中,這些存儲區是按字節來劃分的,每一個存儲單元都能用唯一的 編號去讀或寫數據,這個編號就是常說的存儲單元的地址,而讀寫這個編號的動作就叫做尋 址,通過尋址就能訪問到存儲區中的任一個能訪問的單元,而這個功能是變量或數組等 是不可能代替的。C 語言也因此引入了指針類型的數據類型,專門用來確定其他類型數據的 地址。用一個變量來存放另一個變量的地址,那么用來存放變量地址的變量稱為“指針變量”。 如用變量 STRIP 來存放文章開頭的 STR 變量的地址 51H,變量 STRIP 就是指針變量。下面 用一個圖表來說明變量的指針和指針變量兩個不一樣的概念。 file:///Z:\TEMP\msohtmlclip1\01\clip_image001.png 變量的指針就是變量的地址,用取地址運算符‘&’取得賦給指針變量。&STR 就是把 變量 STR 的地址取得。用語句 STRIP = &STR 就能把所取得的 STR 指針存放在 STRIP 指 針變量中。STRIP 的值就變為 51H。可見指針變量的內容是另一個變量的地址,地址所屬的 變量稱為指針變量所指向的變量。 要訪問變量 STR 除了能用‘STR’這個變量名來訪問之外,還能用變量地址來訪 問。方法是先用&STR 取變量地址并賦于 STRIP 指針變量,然后就能用*STRIP 來對 STR 進行訪問了。‘*’是指針運算符,用它能取得指針變量所指向的地址的值。在上圖中指針 變量 STRIP 所指向的地址是 51H,而 51H 中的值是 40H,那么*STRIP 所得的值就是 40H。 使用指針變量之前也和使用其它類型的變量那樣要求先定義變量,而且形式也相類似, 一般的形式如下: 數據類型 [存儲器類型] * 變量名; unsigned char xdata *pi //指針會占用二字節,指針自身存放在編譯器默認存儲區,指 向 xdata 存儲區的 char 類型 unsigned char xdata * data pi; //除指針自身指定在 data 區,其它同上 int * pi; //定義為一般指針,指針自身存放在編譯器默認存儲區,占三個字節 在定義形式中“數據類型”是指所定義的指針變量所指向的變量的類型。“存儲器類型” 是編譯器編譯時的一種擴展標識,它是可選的。在沒有“存儲器類型”選項時,則定義為一 般指針,如有“存儲器類型”選項時則定義為基于存儲器的指針。限于 51 芯片的尋址范圍, 指針變量最大的值為 0xFFFF,這樣就決定了一般指針在內存會占用 3 個字節,第一字節存 放該指針存儲器類型編碼,后兩個則存放該指針的高低位址。而基于存儲器的指針因為不用 識別存儲器類型所以會占一或二個字節,idata,data,pdata 存儲器指針占一個字節,code,xdata 則會占二個字節。由上可知,明確的定義指針,能節省存儲器的開銷,這在嚴格要求程序 體積的項目中很有用處。 指針的使用方法很多,限于篇幅以上只能對它做一些基礎的介紹。下面用在講述常量時 的例程改動一下,用以說明指針的基本使用方法。 #include //預處理文件里面定義了特殊寄存器的名稱如 P1 口定義為 P1 void main(void) { //定義花樣數據,數據存放在片內 CODE 區中 unsigned char code design[]={0xFF,0xFE,0xFD,0xFB,0xF7,0xEF,0xDF,0xBF,0x7F, 0x7F,0xBF,0xDF,0xEF,0xF7,0xFB,0xFD,0xFE,0xFF, 0xFF,0xFE,0xFC,0xF8,0xF0,0xE0,0xC0,0x80,0x0, 0xE7,0xDB,0xBD,0x7E,0xFF}; unsigned int a; //定義循環用的變量 unsigned char b; unsigned char code * dsi; //定義基于 CODE 區的指針 do{ dsi = &design[0]; //取得數組第一個單元的地址 for (b=0; b<32; b++) { } }while(1); } for(a=0; a<30000; a++); //延時一段時間 P1 = *dsi; //從指針指向的地址取數據到 P1 口 dsi++; //指針加一, 為了能清楚的了解指針的工作原理,能使用 keil uv2 的軟件仿真器查看各變量和存儲器的 值。編譯程序并執行,然后打開變量窗口,如圖。用單步執行,就能查到到指針的變量。 如圖中所示的是程序中循環執行到第二次,這個時候指針 dsi 指向 c:0x0004 這個地址,這個地址 的值是 0xFE。在存儲器窗口則能察看各地址單元的值。使用這種方法不但在學習時能 幫助更好的了解語法或程序的工作,而且在實際使用中更能讓你更快更準確的編寫程序或解 決程序中的問題。 |
|
| file:///Z:\TEMP\msohtmlclip1\01\clip_image003.png |
file:///Z:\TEMP\msohtmlclip1\01\clip_image004.png | file:///Z:\TEMP\msohtmlclip1\01\clip_image002.png | file:///Z:\TEMP\msohtmlclip1\01\clip_image005.png | |
| |||
歡迎光臨 (http://www.raoushi.com/bbs/) | Powered by Discuz! X3.1 |