當前位置:首頁 > 嵌入式 > 嵌入式軟件
[導讀]單片機C語言教程:C51數組的使用

數組不過就是同一類型變量的有序集合。形象的能這樣去理解,就像一個 學校在操場上排隊,每一個級代表一個數據類型,每一個班級為一個數組,每一個學生就是 數組中的一個數據。數據中的每個數據都能用唯一的下標來確定其位置,下標能是一維 或多維的。就如在學校的方隊中要找一個學生,這個學生在 I 年級 H 班 X 組 Y 號的,那么 能把這個學生看做在 I 類型的 H 數組中(X,Y)下標位置中。數組和普通變量一樣,要

求先定義了才能使用,下面是定義一維或多維數組的方式:

“數據類型”是指數組中的各數據單元的類型,每個數組中的數據單元只能是同一數據

類型。“數組名”是整個數組的標識,命名方法和變量命名方法是一樣的。在編譯時系統(tǒng)會 根據數組大小和類型為變量分配空間,數組名能說就是所分配空間的首地址的標識。“常 量表達式”是表示數組的長度和維數,它必須用“[]”括起,括號里的數不能是變量只能是 常量。

unsigned int xcount [10]; //定義無符號整形數組,有 10 個數據單元

char inputstring [5]; //定義字符形數組,有 5 個數據單元

float outnum [10],[10];//定義浮點型數組,有 100 個數據單元

在 C 語言中數組的下標是從 0 開始的而不是從 1 開始,如一個具有 10 個數據單元的數

組 count,它的下標就是從 count[0]到 count[9],引用單個元素就是數組名加下標,如 count[1] 就是引用 count 數組中的第 2 個元素,如果錯用了 count[10]就會有錯誤出現(xiàn)了。還有一點要 注意的就是在程序中只能逐個引用數組中的元素,不能一次引用整個數組,但是字符型的數 組就能一次引用整個數組。

數組也是能賦初值的。在上面介紹的定義方式只適用于定義在內存 DATA 存儲器使 用的內存,有的時候我們需要把一些數據表存放在數組中,通常這些數據是不用在程序中改 變數值的,這個時候就要把這些數據在程序編寫時就賦給數組變量。因為 51 芯片的片內 RAM 很有限,通常會把 RAM 分給參與運算的變量或數組,而那些程序中不變數據則應存放在片 內的 CODE 存儲區(qū),以節(jié)省寶貴的 RAM。賦初值的方式如下:

數據類型 [存儲器類型] 數組名 [常量表達式] = {常量表達式};

數據類型 [ 存儲器類型] 數組名 [ 常量表達式 1]...... [ 常量表達式 N]={{ 常量表達 式}...{常量表達式 N}};

在定義并為數組賦初值時,開始學習的朋友一般會搞錯初值個數和數組長度的關系,而致使 編譯出錯。初值個數必須小于或等于數組長度,不指定數組長度則會在編譯時由實際的初值 個數自動設置。

unsigned char LEDNUM[2]={12,35}; //一維數組賦初值

int Key[2][3]={{1,2,4},{2,2,1}}; //二維數組賦初值

unsigned char IOStr[]={3,5,2,5,3}; //沒有指定數組長度,編譯器自動設置

unsigned char code skydata[]={0x02,0x34,0x22,0x32,0x21,0x12}; //數據保存在 code 區(qū)

下面的一個簡單例子是對數組中的數據進行排序,使用的是冒泡法,一來了解數組的使 用,二來掌握基本的排序算法。冒泡排序算法是一種基本的排序算法,它每次順序取數組中 的兩個數,并按需要按其大小排列,在下一次循環(huán)中則取下一次的一個數和數組中下一個數 進行排序,直到數組中的數據全部排序完成。

#include

#include

void taxisfun (int taxis2[])

{

unsigned char TempCycA,TempCycB,Temp;

for (TempCycA=0; TempCycA<=8; TempCycA++)

for (TempCycB=0; TempCycB<=8-TempCycA; TempCycB++)

{//TempCycB<8-TempCycA 比用 TempCycB<=8 少用很多循環(huán)

if (taxis2[TempCycB+1]>taxis2[TempCycB]) //當后一個數大于前一個 數

{

Temp = taxis2[TempCycB]; //前后 2 數交換

taxis2[TempCycB] = taxis2[TempCycB+1];

taxis2[TempCycB+1] = Temp; //因函數參數是數組名調用形

參的變動影響實參

}

}

}

void main(void)

{

int taxis[] = {113,5,22,12,32,233,1,21,129,3};

char Text1[] = {"source data:"}; //"源數據"

char Text2[] = {"sorted data:"}; //"排序后數據"

unsigned char TempCyc;

SCON = 0x50; //串行口方式 1,允許接收

TMOD = 0x20; //定時器 1 定時方式 2

TCON = 0x40; //設定時器 1 開始計數

TH1 = 0xE8; //11.0592MHz 1200 波特率

TL1 = 0xE8; TI = 1;

TR1 = 1; //啟動定時器

printf("%sn",Text1); //字符數組的整體引用

for (TempCyc=0; TempCyc<10; TempCyc++)

printf("%d ",taxis[TempCyc]);

printf("n----------n");

taxisfun (taxis); //以實際參數數組名 taxis 做參數被函數調用

printf("%sn",Text2);

for (TempCyc=0; TempCyc<10; TempCyc++) //調用后 taxis 會被改變

printf("%d ",taxis[TempCyc]);

while(1);

}

例子中能看出,數組同樣能作為函數的參數進行傳遞。數組做參數時是用數組名進行傳遞的,一個數組的數組名表示該數組的首地址,在用數組名作為函數的調用參數時,它 的傳遞方式是采用了地址傳遞,就是將實際參數數組的首地址傳遞給函數中的形式參數數 組,這個時候實際參數數組和形式參數數組實際上是使用了同一段內存單元,當形式參數數組在 函數體中改變了元素的值,同時也會影響到實際參數數組,因為它們是存放在同一個地址的。 上面的例子同時還使用到字符數組。字符數組中每一個數據都是一個字符,這樣一個一 維的字符數組就組成了一個字符串,在 C 語言中字符串是以字符數組來表達處理的。為了 能測定字符串的長度,C 語言中規(guī)定以‘o’來做為字符串的結束標識,編譯時會自動在字 符串的最后加入一個‘o’,那么要注意的是如果用一個數組要保存一個長度為 10 字節(jié)的字 符串則要求這個數組至少能保存 11 個元素。‘o’是轉義字符,它的含義是空字符,它的 ASCII 碼為 00H,也就是說當每一個字符串都是以數據 00H 結束的,在程序中操作字符數 據組時要注意這一點。字符數組除了能對數組中單個元素進行訪問,還能訪問整個數組, 其實整個訪問字符數組就是把數組名傳到函數中,數組名是一個指向數據存放空間的地址指 針,函數根據這個指針和‘/o’就能完整的操作這個字符數組。對于這一段所說的,能 參看下面一例 1602LCD 顯示模塊的驅動演示例子進行理解。這里要注意就是能用單個字符數組元素來進行運算,但不能用整個數組來做運算,因為數組名是指針而不是數據。

[!--empirenews.page--]

/*============================================================

使用 1602 液晶顯示的實驗例子

==============================================================

SMC1602A(16*2)模擬口線接線方式 連接線圖:

---------------------------------------------------

|LCM-----51 | LCM-----51 | LCM------51 |

---------------------------------------------|

|DB0-----P1.0 | DB4-----P1.4 | RW-------P2.0 |

|DB1-----P1.1 | DB5-----P1.5 | RS-------P2.1 |

|DB2-----P1.2 | DB6-----P1.6 | E--------P2.2 |

|DB3-----P1.3 | DB7-----P1.7 | VLCD 接 1K 電阻到 GND|

---------------------------------------------------

[注:AT89S51 使用 12M 晶體震蕩器]

=============================================================*/

#define LCM_RW P2_0 //定義引腳

#define LCM_RS P2_1

#define LCM_E P2_2

#define LCM_Data P1

#define Busy 0x80 //用于檢測 LCM 狀態(tài)字中的 Busy 標識

#include

void WriteDataLCM(unsigned char WDLCM);

void WriteCommandLCM(unsigned char WCLCM,BuysC);

unsigned char ReadDataLCM(void); unsigned char ReadStatusLCM(void); void LCMInit(void);

void DisplayOneChar(unsigned char X, unsigned char Y, unsigned char DData);

void DisplayListChar(unsigned char X, unsigned char Y, unsigned char code *DData);

void Delay5Ms(void);

void Delay400Ms(void);

unsigned char code cdle_net[] = {"www.xx.com"};

unsigned char code email[] = {"pnzwzw@xx.com"};

void main(void)

{

Delay400Ms(); //啟動等待,等 LCM 講入工作狀態(tài)

LCMInit(); //LCM 初始化

Delay5Ms(); //延時片刻(可不要)

DisplayListChar(0, 0, cdle_net); DisplayListChar(0, 1, email); ReadDataLCM();//測試用句無意義 while(1);

}

//寫數據

void WriteDataLCM(unsigned char WDLCM)

{

ReadStatusLCM(); //檢測忙 LCM_Data = WDLCM; LCM_RS = 1;

LCM_RW = 0;

LCM_E = 0; //若晶體震蕩器速度太高能在這后加小的延時

LCM_E = 0; //延時

LCM_E = 1;

}

//寫指令

void WriteCommandLCM(unsigned char WCLCM,BuysC) //BuysC 為 0 時忽略忙檢測

{

if (BuysC) ReadStatusLCM(); //根據需要檢測忙

LCM_Data = WCLCM; LCM_RS = 0; LCM_RW = 0;

LCM_E = 0;

LCM_E = 0; LCM_E = 1;

}

//讀數據

unsigned char ReadDataLCM(void)

{

LCM_RS = 1; LCM_RW = 1; LCM_E = 0; LCM_E = 0; LCM_E = 1; return(LCM_Data);

}

//讀狀態(tài)

unsigned char ReadStatusLCM(void)

{

LCM_Data = 0xFF; LCM_RS = 0; LCM_RW = 1; LCM_E = 0; LCM_E = 0; LCM_E = 1;

while (LCM_Data & Busy); //檢測忙信號

return(LCM_Data);

}

void LCMInit(void) //LCM 初始化

{

LCM_Data = 0;

WriteCommandLCM(0x38,0); //三次顯示模式設置,不檢測忙信號

Delay5Ms(); WriteCommandLCM(0x38,0); Delay5Ms(); WriteCommandLCM(0x38,0); Delay5Ms();

WriteCommandLCM(0x38,1); //顯示模式設置,開始要求每次檢測忙信號

WriteCommandLCM(0x08,1); //關閉顯示 WriteCommandLCM(0x01,1); //顯示清屏 WriteCommandLCM(0x06,1); // 顯示光標移動設置 WriteCommandLCM(0x0C,1); // 顯示開及光標設置

}

//按指定位置顯示一個字符

void DisplayOneChar(unsigned char X, unsigned char Y, unsigned char DData)

{

Y &= 0x1;

X &= 0xF; //限制 X 不能大于 15,Y 不能大于 1

if (Y) X |= 0x40; //當要顯示第二行時地址碼+0x40; X |= 0x80; //算出指令碼

WriteCommandLCM(X, 0); //這里不檢測忙信號,發(fā)送地址碼

WriteDataLCM(DData);

}

//按指定位置顯示一串字符

void DisplayListChar(unsigned char X, unsigned char Y, unsigned char code *DData)

{

unsigned char ListLength;

ListLength = 0; Y &= 0x1;

X &= 0xF; //限制 X 不能大于 15,Y 不能大于 1

while (DData[ListLength]>0x20) //若到達字串尾則退出

{

if (X <= 0xF) //X 坐標應小于 0xF

{

DisplayOneChar(X, Y, DData[ListLength]); //顯示單個字符

ListLength++; X++;

}

}

}

//5ms 延時

void Delay5Ms(void)

{

unsigned int TempCyc = 5552;

while(TempCyc--);

}

//400ms 延時

void Delay400Ms(void)

{

unsigned char TempCycA = 5; unsigned int TempCycB; while(TempCycA--)

{

TempCycB=7269;

while(TempCycB--);

};

}

本站聲明: 本文章由作者或相關機構授權發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內容真實性等。需要轉載請聯(lián)系該專欄作者,如若文章內容侵犯您的權益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或將催生出更大的獨角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數字化轉型技術解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關鍵字: AWS AN BSP 數字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術公司SODA.Auto推出其旗艦產品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關鍵字: 汽車 人工智能 智能驅動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務能7×24不間斷運行,同時企業(yè)卻面臨越來越多業(yè)務中斷的風險,如企業(yè)系統(tǒng)復雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務連續(xù)性,提升韌性,成...

關鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據媒體報道,騰訊和網易近期正在縮減他們對日本游戲市場的投資。

關鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數據產業(yè)博覽會開幕式在貴陽舉行,華為董事、質量流程IT總裁陶景文發(fā)表了演講。

關鍵字: 華為 12nm EDA 半導體

8月28日消息,在2024中國國際大數據產業(yè)博覽會上,華為常務董事、華為云CEO張平安發(fā)表演講稱,數字世界的話語權最終是由生態(tài)的繁榮決定的。

關鍵字: 華為 12nm 手機 衛(wèi)星通信

要點: 有效應對環(huán)境變化,經營業(yè)績穩(wěn)中有升 落實提質增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務引領增長 以科技創(chuàng)新為引領,提升企業(yè)核心競爭力 堅持高質量發(fā)展策略,塑強核心競爭優(yōu)勢...

關鍵字: 通信 BSP 電信運營商 數字經濟

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術學會聯(lián)合牽頭組建的NVI技術創(chuàng)新聯(lián)盟在BIRTV2024超高清全產業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術創(chuàng)新聯(lián)...

關鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(集團)股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關鍵字: BSP 信息技術
關閉
關閉