版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進(jìn)行舉報(bào)或認(rèn)領(lǐng)
文檔簡介
5.1C51基本語法規(guī)則
5.2C51程序設(shè)計(jì)技巧
5.3匯編語言與C語言的混合編程
5.4C51與匯編語言的對照
5.5C51程序設(shè)計(jì)小結(jié)
習(xí)題五第五章C51程序設(shè)計(jì)語言5.1.1C51數(shù)據(jù)類型
C51支持的數(shù)據(jù)類型如表5-1所示。標(biāo)準(zhǔn)C語言和C51相同的數(shù)據(jù)類型不再詳細(xì)說明,下面主要解釋C51擴(kuò)展的特殊
數(shù)據(jù)類型。5.1C51基本語法規(guī)則表5-1C51支持的數(shù)據(jù)類型5.1.2常量與變量
1.常量
常量就是程序運(yùn)行過程中不能改變值的量,可用在值不必改變的場合,如固定的數(shù)據(jù)表、字庫等。
常量的定義方式有如下幾種:
(1)用預(yù)定義語句定義常量,如:
#difineFalse0x0;//定義False為0
(2)用code語句定義常量,如:
unsignedintcodea=100;//定義a為無符號int常量100
(3)用const語句定義常量,如:
constunsignedintc=100;//定義c為無符號int常量100
常量的數(shù)據(jù)類型有整型、浮點(diǎn)型、字符型、字符串型和位標(biāo)量,這與標(biāo)準(zhǔn)C語言的常量一致。
2.變量
定義一個變量的格式如下:
[存儲種類]數(shù)據(jù)類型[存儲器類型]變量名稱
表5-2給出了C51編譯器能夠識別的存儲器類型,變量名稱的命名規(guī)則與標(biāo)準(zhǔn)C語言一致,下面給出了一些變量定義的例子:chardatavar1 ;在data區(qū)定義字符型變量var1
intidatavar2 ;在data區(qū)定義字符型變量var2
inta=5 ;定義變量a,同時賦以初值5,變量a位
;于由編譯模式確定的默認(rèn)存儲區(qū)
externfloatidatax,y,z;在idata區(qū)定義外部浮點(diǎn)型變量x,y,z表5-2C51能夠識別的存儲器類型
C51編譯器有三種存儲模式,具體如下:
(1)小(small)模式
(2)緊湊(compact)模式
(3)大(large)模式
1)全局變量
全局變量是指在程序開始處或各個功能函數(shù)的外面定義的變量。全局變量在整個程序的執(zhí)行過程中都要占用內(nèi)存單元。在程序開始處定義的全局變量對于整個程序都有效,可供程序中所有函數(shù)共同使用。例如:#include〈reg51.h〉 ;頭文件,定義單片機(jī)片內(nèi)資源#defineucharunsignedchar ;定義常量uchar=unsignedchar
ucharkey ;定義全局變量key
voidmain()
{
}在各功能函數(shù)外面定義的全局變量只對從定義處開始往后的各個函數(shù)有效。若一個全局變量不是在程序文件開始處定義的,但又希望在它的定義點(diǎn)之前的函數(shù)中引用該變量,這時應(yīng)在引用該變量的函數(shù)中用關(guān)鍵字extern將其說明為外部變量。特別是當(dāng)一個程序能由多個源程序文件組成時,如果一個程序中需要引用另外一個文件中已經(jīng)定義的外部變量,必須要使用extern來聲明。例如:文件1:
#include〈reg51.h〉
unsignedintarray[10];定義全局變量array
voidfillarray()
voidinit_ser()
{
}
voidmain()
{
}文件2:
externintarray[10];在另外一個文件中引用變量array
voidfillarray()
{
}
2)局部變量
下面給出了一個局部變量聲明的例子:
voidfillarray()
{
intarray[10];定義局部array
}5.1.3運(yùn)算符與表達(dá)式
C51的運(yùn)算符和表達(dá)式與標(biāo)準(zhǔn)C語言差別不大,表5-3總結(jié)性地給出了運(yùn)算符及其在表達(dá)式中的優(yōu)先級關(guān)系,供讀者參考。表5-3C51支持的運(yùn)算符及其優(yōu)先級5.1.4程序控制語句
1.表達(dá)式語句
表達(dá)式語句是一種最基本的語句。C51語言中,在表達(dá)式右邊加一個分號“;”就構(gòu)成了表達(dá)式語句,下面的語句都是合法的表達(dá)式語句:
b=b*10;Count++;
X=A;Y=B;
Page=(a+b)/a-1;在C51語言中有一個特殊的表達(dá)式語句,稱為空語句,它僅僅由一個分號“;”組成。有時候?yàn)榱耸拐Z法正確,但并不要求有具體的動作,這時就可以采用空語句。例如:
repeat:; //定義標(biāo)號
gotorepeat; //無條件跳轉(zhuǎn)到標(biāo)號repeat處
上述程序利用標(biāo)號repeat和無條件轉(zhuǎn)移指令goto語句構(gòu)成循環(huán),但在標(biāo)號repeat處并不需要有具體的動作??照Z句在循環(huán)中也經(jīng)常用到。
2.復(fù)合語句
若干條表達(dá)式語句組合而成的語句稱為復(fù)合語句,其一般形式為:
{
局部變量定義
語句1;
語句2;
語句n;
}復(fù)合語句通常出現(xiàn)在函數(shù)中。下面給出了一個復(fù)合語句短句的簡單例子:
{
intx1,y1; //定義局部變量
x1=5;
y1=8;
printf(″%d%d″,x1,y1); //打印x1,y1
}
3.條件語句
C51語言提供了三種形式的條件語句:
(1)基本形式,其語法結(jié)構(gòu)為:
if(表達(dá)式)語句:
語義:如果表達(dá)式的值為真,則執(zhí)行其后的語句,否則不執(zhí)行該語句。
(2)if-else形式,其語法結(jié)構(gòu)為:
if(表達(dá)式)語句1;
else語句2;
語義:如果表達(dá)式的值為真,則執(zhí)行語句1,否則執(zhí)行語句2。
(3)if-else-if形式,其語法結(jié)構(gòu)為:
if(表達(dá)式1)語句1;
elseif(表達(dá)式2) 語句2;
elseif(表達(dá)式3) 語句3;
elseif(表達(dá)式m) 語句m;
else 語句n;
語義:依次判斷表達(dá)式的值,當(dāng)出現(xiàn)某個值為真時,則執(zhí)行其對應(yīng)的語句。然后跳到整個if語句之外繼續(xù)執(zhí)行程序。如果所有的表達(dá)式均為假,則執(zhí)行語句n。下面給出了一個簡單的條件語句應(yīng)用的例子:
voidmain()
{
Inta,b; //定義變量
Printf(″inputtwonumbers:″)
Scanf(″%d%d″,&a,&b);//從外部輸入兩個數(shù)據(jù)
If(a>b)printf(″max=%d\n″,a) //比較大小,并
//打印最大值
elseprintf(″max=%d\n″,b);
}
4.開關(guān)語句
開關(guān)語句是用關(guān)鍵字switch構(gòu)成的,它的一般形式如下:
switch(表達(dá)式)
{
case常量表達(dá)式1;語句1;
break;
case常量表達(dá)式2;語句2;
break;
case常量表達(dá)式n;語句n;
break;
default:語句n+1;
}
語義:計(jì)算表達(dá)式的值,并逐個與其后的常量表達(dá)式的值相比較,當(dāng)表達(dá)式的值與某個常量表達(dá)式的值相等時,則執(zhí)行case后面的語句,再執(zhí)行break(間斷語句)語句,跳出switch語句。如果case后沒有和條件相等的值時就執(zhí)行default后的語句。當(dāng)要求沒有符合的條件時不做任何處理,則不寫default語句。
使用switch語句時要注意,在case后的各常量表達(dá)式的值不能相同,否則會出現(xiàn)錯誤。在case后,允許有多個語句,可以不用“{}”括起來。各case和default子句的先后順序可
以變動,而不會影響程序執(zhí)行結(jié)果。voidmain()
{
inta;
printf(″inputintegernumber:″);
scanf(″%d″,&a); //從外部輸入一個數(shù)據(jù)
switch(a)
{
case1:printf(″Monday\n″);break;
//如果該數(shù)據(jù)為1,打印星期一,退出switch語
case2printf(″Tuesday\n″);break;
//如果該數(shù)據(jù)為2,打印星期二,退出switch語句
case3:printf(″Wednesday\n″);break;
//如果該數(shù)據(jù)為3,打印星期三,退出switch語句
case4:printf(″Thursday\n″);break;
//如果該數(shù)據(jù)為4,打印星期四,退出switch語句
case5:printf(″Firday\n″);break;
//如果該數(shù)據(jù)為5,打印星期五,退出switch語句
case6:printf(″Saturday\n″);break;
//如果該數(shù)據(jù)為6,打印星期六,退出switch語句
case1:printf(″Sunday\n″);break;
//如果該數(shù)據(jù)為7,打印星期日,退出switch語句
default:printf(″error\n″); //否則,打印出錯信息
}
}
5.循環(huán)語句
提供了四種循環(huán)語句:
(1)while語句,其語法結(jié)構(gòu)為:
while(表達(dá)式)語句;
(2)do-while語句,其語法結(jié)構(gòu)為:
do
語句;
while(表達(dá)式);
(3)for語句,for語句是C51語言所提供的功能更強(qiáng),使用更廣泛的一種循環(huán)語句,其語法結(jié)構(gòu)為:
for(表達(dá)式1;表達(dá)式2;表達(dá)式3)
語句;
下面給出了一個循環(huán)語句應(yīng)用的簡單例子,完成1~10的累加并通過串口顯示:
#includde〈AT89X51.H〉
#include〈stdio,h〉
intcount(void);//聲明函數(shù)
voidmain(void){
unsignedinttemp;
SCON=0x50; //串口方式1,允許接收TMOD=0x20;定時器1定時方式2
TCON=0x40; //設(shè)置定時器1開始計(jì)數(shù)
TH1=0xE8; //晶振11.0592MHz,1200波特率,TL1=0xE8;
TI=1;
TR1=1; //啟動定時器
Temp=Count();
printf(11-10SUM=%d\n11,temp);//顯示
while(1);
}
intcount(void)
{
unsignedintI,SUM;
for(I=1;I<=10;I++)
{
SUM=I+SUM; //累加
}
return(SUM);
}
6.轉(zhuǎn)移語句
(1)goto語句。goto語句也稱為無條件轉(zhuǎn)移語句,其語法結(jié)構(gòu)為:
goto語句標(biāo)號;
(2)break語句。break語句只能用在switch語句或循環(huán)語句中,其語法結(jié)構(gòu)為:
break;
(3)continue語句。continue語句只能用在循環(huán)體中,其語法結(jié)構(gòu)為:
continue;
(4)return語句。return語句是返回語句,其語法結(jié)構(gòu)為:
return(表達(dá)式);
或
return;5.1.5函數(shù)
1.函數(shù)的調(diào)用
(1)函數(shù)語句。在主調(diào)用函數(shù)中直接將被調(diào)用函數(shù)作為一個語句。如:
Print(″HelloWord!\n″);
(2)函數(shù)表達(dá)式。這種調(diào)用方式是將被調(diào)用函數(shù)作為一個表達(dá)式,這種表達(dá)式稱為函數(shù)表達(dá)式,如:
c=power(x,n)+power(y,m);
(3)函數(shù)參數(shù)。這種參數(shù)調(diào)用方式是指將被調(diào)用函數(shù)作為另一個函數(shù)的實(shí)際參數(shù),如:
M=jiec(jiec(3));
2.被調(diào)用函數(shù)的說明
與使用變量一樣,在調(diào)用一個函數(shù)之前,包括標(biāo)準(zhǔn)庫函數(shù),必須對該函數(shù)的類型進(jìn)行說明,即“先說明,后調(diào)用”。如果調(diào)用的是標(biāo)準(zhǔn)庫函數(shù),一般應(yīng)在程序的開始處用預(yù)處理命令#include將有關(guān)函數(shù)說明的頭文件包含進(jìn)來,如:
include〈reg51.h〉如果調(diào)用的是用戶自定義函數(shù),而且該函數(shù)與調(diào)用它的主調(diào)用函數(shù)在同一個文件中,一般應(yīng)在主調(diào)用函數(shù)中對被調(diào)用函數(shù)的類型進(jìn)行說明。函數(shù)說明的一般形式為:
類型標(biāo)識符函數(shù)的名稱(形式參數(shù)表);
其中,“類型標(biāo)識符”說明了函數(shù)返回值的類型,“形式參數(shù)表”說明了各個形式參數(shù)的類型。
例如:
intcount(void);
3.函數(shù)的返回值
在調(diào)用函數(shù)過程中,經(jīng)常希望得到一個從被調(diào)用函數(shù)中帶回來的值,這就是函數(shù)的返回值。
4.參數(shù)的傳遞
(1)數(shù)值傳遞
(2)地址傳遞
5.重入函數(shù)
重入函數(shù)聲明的關(guān)鍵字為reentrant,重入函數(shù)的聲明的格式為:
函數(shù)類型函數(shù)名(形式參數(shù)表)reentrant下面給出了一個簡單重入函數(shù)定義的例子:
charmax(chara,charb,charc)reentrant
{
charmax_data;
max_data=(a>b)?a:b; //將a,b兩數(shù)中的最大
//值賦給max_data
max_data=(max_data>c)?max_data:c; //將a,b,c三數(shù)中的
//最大值賦予maxdata
return(max_data); //返回最大值
}
6.中斷服務(wù)函數(shù)
中斷服務(wù)函數(shù)聲明的關(guān)鍵字為interrupt,聲明的格式為:
函數(shù)類型函數(shù)名(形式參數(shù))interruptn[usingn]
其中,關(guān)鍵字interrupt后面的n是中斷號,n的取值范圍為0~31。編譯器從8n+3處產(chǎn)生中斷向量,具體的中斷號n和中斷向量取決于8051系列單片機(jī)芯片的型號,常用中斷源和中斷向量。如表5-4所示。表5-4單片機(jī)常用中斷號與中斷向量下面給出了一個簡單的中斷服務(wù)函數(shù)的例子:
unsignedintg_wTCount;
unsignedcharg_bySecond;
voidtimer0(void)interrupt1using1
{
g_wTCount++
if(g_wTCount>=20 //如果計(jì)數(shù)到20
{
g_bySecond++; //另一個計(jì)數(shù)器
g_wTCount=0; //計(jì)數(shù)器清零
}
}5.1.6指針
圖5-1形象地給出看了指針變量和它所指向的變量之間的關(guān)系。圖5-1指針變量和它所指向的變量
2.指針變量的定義
指針變量的定義與一般變量的定義類似,其一般形式如下:
數(shù)據(jù)類型[存儲器類型1]*[存儲器類型2]標(biāo)識符;
(1)通用指針變量:不選用“存儲器類型1”選項(xiàng)的指針變量稱為通用指針變量,其聲明和標(biāo)準(zhǔn)C語言中一樣。如:
char*s;
int*numptr;
long*state;
(2)存儲器指針變量:選用“存儲器類型1”選項(xiàng)的指針變量稱為存儲器指針變量,如:
chardata*str;
intxdata*numtab;
longcode*powtab;下面的示例程序說明了使用不同的指針在代碼長度、占用數(shù)據(jù)空間和運(yùn)行時間上的不同:
Description IdataPointer XdataPointer
GenericPointer
C源程序 idata*ip; charxdata*xp;
char*p;
charval; charval;
charval;
val=*ip; val=*xp;
val=*xp;
編譯后的代碼 MOVR0,ip MOVDPL,xp+1
MOVR1,p+2
MOVval,@R0 MOVDPH,xpMOVR2,p+1
MOVA,@DPTRMOVR3,p
MOVval,ACALLCLDPTR
指針大小 1byte 2byte
3byte
代碼長度 4byte 9byte
11byte+librarycall
執(zhí)行時間 4cycles 7cycles 13cycles由于許多庫函數(shù)會使用通用指針,但為了提高效率,用戶在程序中其自定義的函數(shù)可能會使用存儲器指針。在不同指針的函數(shù)之間進(jìn)行調(diào)用時,往往要進(jìn)行指針的轉(zhuǎn)換。指針的轉(zhuǎn)換可以用類型轉(zhuǎn)換的直接程序代碼來強(qiáng)迫轉(zhuǎn)換,或在編譯器內(nèi)部強(qiáng)制轉(zhuǎn)換。C51編譯器指針轉(zhuǎn)換的規(guī)則如表5-5所示。表5-5通用指針與存儲器指針之間的轉(zhuǎn)換
3.指針變量的操作
(1)取地址運(yùn)算符&:該運(yùn)算符用來取變量的地址。如取變量a地址的指令為&a。
(2)間接訪問運(yùn)算符*:該運(yùn)算符用來間接訪問變量。如*p為指針變量p所指向的變量。
指針變量經(jīng)過定義后可以像其他基本變量一樣引用,下面的例子給出了指針變量基本操作:
inti,x,y,*pi,*px //定義指針變量
pi=&i;
//指針變量賦值將變量i的地址賦
//給指針變量pi,使pi指向i
*pi+=1 //指針變量的運(yùn)算,等價于i+=1
下面是一個指針應(yīng)用的簡單例子:
#include〈reg51.H〉
#inculde〈stdio.h〉
voidmain(void)
{
intx,y; //定義一般變量
int*p,*p1,*p2; //定義指針變量
printf(″inputxandy:\n″);
sacnf(″%d%d,&x,&y″); //輸入兩個數(shù)
p1=&x;p2=&y;
if(x<y)
{
p=p1;p1=p2;p2=p; //將大的數(shù)放在指針p1指向的變量,小
//的數(shù)放在p2指向的變量
}
printf(″max=%d,min=%d,\n″,*p1,*p2);//打印結(jié)果
while(1);
}5.1.7構(gòu)造數(shù)據(jù)類型
1.數(shù)組
數(shù)組是一組有序數(shù)據(jù)的集合,數(shù)組中的每一個數(shù)據(jù)都屬于同一種數(shù)據(jù)類型。數(shù)組中的各個元素可以用數(shù)組名和下標(biāo)來唯一地確定。按照維數(shù),數(shù)組可以分為一維數(shù)組和多維數(shù)組。在C51中數(shù)組必須先定義,然后才能使用。一維數(shù)組的定義形式如下:
數(shù)據(jù)類型數(shù)組名[常量表達(dá)式];定義多維數(shù)組時,只要在數(shù)組名后面增加相應(yīng)于維數(shù)的常量表達(dá)式即可,其一般形式如下:
數(shù)據(jù)類型數(shù)組名[常量表達(dá)式1]…[常量表達(dá)式N];
下面給出了幾個數(shù)組定義的例子:
unsignedintxcount[10];
//定義一堆無符號整型數(shù)組,有10個數(shù)據(jù)單元
charinputstring[5];
//定義一維字符型數(shù)組,有5個數(shù)據(jù)單元
floatoutnum[10],[10];
//定義二維浮點(diǎn)型數(shù)組,有100個數(shù)據(jù)單元
2.結(jié)構(gòu)
結(jié)構(gòu)是由基本數(shù)據(jù)類型構(gòu)成的,并用一個標(biāo)識符來命名的各種變量的組合。結(jié)構(gòu)中可以使用不同的數(shù)據(jù)類型。結(jié)構(gòu)也是一種數(shù)據(jù)類型,可以使用結(jié)構(gòu)變量,因此,像其他類型的變量一樣,在使用結(jié)構(gòu)變量時要先對其定義。定義結(jié)構(gòu)變量的一般格式為:
struct結(jié)構(gòu)名
{
數(shù)據(jù)類型變量名;
數(shù)據(jù)類型變量名;
…
}
結(jié)構(gòu)變量;構(gòu)成結(jié)構(gòu)的每一個類型變量稱為結(jié)構(gòu)成員,它像數(shù)組的元素一樣,但數(shù)組中元素是以下標(biāo)來訪問的,而結(jié)構(gòu)是按變量名字來訪問成員的。下面的例子定義了一個結(jié)構(gòu)名為string的結(jié)構(gòu)變量person:
structstring
{
charname[8];
intage;
charsex[2];
chardepart[20];
floatwage1,wage2,wage3,wage4,wage5;
}
person;結(jié)構(gòu)是一個新的數(shù)據(jù)類型,因此結(jié)構(gòu)變量也可以像其他類型的變量一樣賦值、運(yùn)算,不同的是結(jié)構(gòu)變量以成員作為基本變量。結(jié)構(gòu)成員的表示方式為:
結(jié)構(gòu)變量.成員名
如果將“結(jié)構(gòu)變量.成員名”看成一個整體,則這個整體的數(shù)據(jù)類型與結(jié)構(gòu)中該成員的數(shù)據(jù)類型相同,這樣就可像前面所講的變量那樣使用它們。如對上面例子中的年齡進(jìn)行賦值:
person.age=20;
3.聯(lián)合
聯(lián)合也是一種新的數(shù)據(jù)類型,它是一種特殊形式的變量。聯(lián)合說明和聯(lián)合變量定義與結(jié)構(gòu)十分相似。其形式為:
union聯(lián)合名
{
數(shù)據(jù)類型成員名;
數(shù)據(jù)類型成員名;
…
}
聯(lián)合變量名;聯(lián)合表示幾個變量共用一個內(nèi)存位置,在不同的時間保存不同的數(shù)據(jù)類型和不同長度的變量。下面的例子定義了一個聯(lián)合名為a_bc的聯(lián)合變量lgc:
uniona_bc
{
inti;
charmm;
}
lgc同樣聯(lián)合變量也可以定義成數(shù)組或指針,但當(dāng)定義為指針時,此時聯(lián)合訪問成員可表示成:
聯(lián)合名_成員名
另外,聯(lián)合可以出現(xiàn)在結(jié)構(gòu)內(nèi),它的成員也可以是結(jié)構(gòu)。例如:
struct
{
intage;
char*addr;
union
{
inti;
char*ch;
}
x;
}
y[10];
4.枚舉
枚舉是一個被命名的整型常數(shù)的集合,枚舉的說明與結(jié)構(gòu)的聯(lián)合相似,其形式為:
enum枚舉名
{
標(biāo)示符[=整型常數(shù)],
標(biāo)示符[=整型常數(shù)],
標(biāo)示符[=整型常數(shù)],
}
枚舉變量;若果枚舉沒有初始化,即省掉“=整型常數(shù)”時,則從第一個標(biāo)示符開始,順次賦給標(biāo)示符0,1,2,…。但當(dāng)枚舉中的某個成員賦值后,其后的成員按依次加一的規(guī)則確定其值。例如下列枚舉:
enumstring{x1,x2,x3,x4}x;
說明后,x1,x2,x3,x4的值分別為0,1,2,3,當(dāng)定義改變成:
enumstring
{
x1,
x2=0,
x3=50,
x4,
}
x;5.1.8C51位操作及其表達(dá)式
C51提供了如下位操作運(yùn)算符:
& 按位與
| 按位或
^ 按位異或
~ 按位取反
<< 位左移
>> 位右移
1.“按位與”運(yùn)算符“&”
運(yùn)算規(guī)則:參加運(yùn)算的兩個運(yùn)算對象,若兩者相應(yīng)的位都為1,則該位結(jié)果值為1,否則為0。即
0&0=0
0&1=0
1&0=0
1&1=1
【例5-1】若a=54H=010100B,b=3BH=00111011B,
則表達(dá)式c=a&b的值為10H,即
a:01010100
b:&00111011
(10H)
c:=0010000
2.“按位或”運(yùn)算符“|”
運(yùn)算規(guī)則:參加運(yùn)算的兩個運(yùn)算對象,若兩者相應(yīng)的位置中只要有一個為1,則該位結(jié)果為1。即
0|0=0
0|1=1
1|0=1
1|1=1
【例5-2】若a=30H=00110000B,b=0FH=00001111B,
則表達(dá)式c=a|b的值為3FH,即
A:00110000
B:|
00001111
(3FH)
C=00111111
3.“按位異或”運(yùn)算符“^”
運(yùn)算規(guī)則:參加運(yùn)算的兩個運(yùn)算對象,若兩者相應(yīng)的位置相同,則結(jié)果為0;若兩者相應(yīng)的位置相異,則結(jié)果為1。即
0^0=0
0^1=1
1^0=1
1^1=0
【例5-3】若a=A5H=10100101,b=37H=00110111,
則表達(dá)式c=a^b的值為92H,即
a:10100101
b:00110111
(92H)
10010010
4.“按位取反”運(yùn)算符“~”
~是一個單目運(yùn)算符,用來對一個二進(jìn)制數(shù)按位進(jìn)行取反,即0變1,1變0。
【例5-4】若a=FOH=11110000B,則表達(dá)式a=~a值為0FH,即
a:11110000
~
(0FH)
00001111
5.“位左移”和“位右移”運(yùn)算符“<<”和“>>”
位左移運(yùn)算符<<和位右移運(yùn)算符>>,用來將一個數(shù)各二進(jìn)制位的全部左移或右移若干位,移位后,空白為補(bǔ)0,而溢出位舍棄。
【例5-5】若a=FAH=111010101B,則表達(dá)式a=a<<2將a值左移兩位,其結(jié)果為A8H,即
【例5-6】若a=11000011B=E3H,將a值右循環(huán)位移2位。
a值右循環(huán)位移n位,即將a中原來左邊(8-n)位右移n位,而將原來右邊的n位移到最左面n位。
上述問題可以通過下列步驟來實(shí)現(xiàn):
(1)將a的右端n位先放到b的高n位中,即
b=a<<(8-n);
(2)將a的右端n位,其左邊高n位補(bǔ)0,即
c=a>>n;
(3)將b、c進(jìn)行或運(yùn)算,即
a=c|b;
故對a進(jìn)行循環(huán)右移2位的程序可這樣編寫:
main()
{
unsignedchara=0xc3,b,c;
intn=2;
b=a<<(8-n);
c=a>>n;
a=c|b;
}
結(jié)果:循環(huán)右移前a=11000011
循環(huán)右移后a=11110000
【例5-7】分析下面程序段的作用。
#definePORTAXBYTE[0XFFC0]
voidmain()
{…
PORTA=(PORTA&0xbf)|0x4;
…
}5.1.9自增減運(yùn)算符、復(fù)合運(yùn)算符及其表達(dá)式
1.自增減運(yùn)算符
【例5-8】若i值原來為5,則
j=++i:j值為6,i值也為6
j=i++:j值為5,i值為6
【例5-9】若i原值為3,則表達(dá)式k=(++i)+(++i)+(++i)的值為18。這是因?yàn)?++i最先執(zhí)行,先對表達(dá)式進(jìn)行掃描,對i進(jìn)行3次自加(++i),此時i=6,然后執(zhí)行k=6+6+6=18,故k值為18。
而表達(dá)式k=(i++)+(i++)+(i++)其結(jié)果是:k值為9,而i值為6。因?yàn)橄葘進(jìn)行3次相加,再執(zhí)行3次i的自加。
注意:(1)自增運(yùn)算符(++)和自減運(yùn)算符(--)只能用于變量而不能用于常量表達(dá)式。
(2)(++)和(--)的結(jié)合方向是“自右向左”。
【例5-10】-i++相當(dāng)于-(i++),假如i的原值為3,則表達(dá)式k=-i++的結(jié)果k值為-3,而i值為4。
2.復(fù)合運(yùn)算符及其表達(dá)式
凡是雙目運(yùn)算符,都可以與賦值運(yùn)算符“=”一起組成復(fù)合賦值運(yùn)算符。C51共提供了10種復(fù)合賦值運(yùn)算符。即
+=,-=,*=,/=,%=,<<=,>>=,&=,^=,|=
采用這種復(fù)合賦值運(yùn)算的目的是簡化程序,提高C程序編譯效率。如:a+=b 相當(dāng)于 a=a+b
a-=b 相當(dāng)于 a=a-b
a*=b 相當(dāng)于 a=a*b
a/=b 相當(dāng)于 a=a/b
a%=b 相當(dāng)于 a%b
a<<=8 相當(dāng)于 a=a<<8
a>>8 相當(dāng)于 a=a>>8
5.2.1存取AT89S52單片機(jī)特殊功能寄存器
單片機(jī)AT89S52內(nèi)部有一些特殊功能寄存器(SFR),如P0、P1其對應(yīng)的內(nèi)存地址分別為80H、90H、A0H、B0H……在編寫匯編程序時可以對端口1做輸出控制:
MOVA,#0
MOVP1,A5.2C51程序設(shè)計(jì)技巧這兩條指令將端口1全部設(shè)為低電位。對于C51語言,則可以用以下方式來控制:
#include″AT89S52reg.h″
P1=0;
只要將定義文件AT89S521reg.h包括進(jìn)來即可。因?yàn)轭^文件中已對AT89S52內(nèi)部各個專用寄存器做了定義,其內(nèi)容如下:/*
*AT89S52internalregisterdehnitionsforDDSMICRO-C/51
*
*Copyright1991-1994DaveDunfield
*Allrightsreserved.
*/
externalregisterunsignedcharPSW,SP,DPH,DPL,P0,P1,P2,P3,IP,IE,TMOD,TCON,TH0,TL0,TH1,TL1,SCON,SBUF,PCON,
T2CON,TH2,TL2,RCAP2H,RCAP2L;
若想從端口1中讀取數(shù)據(jù)至變量in中,可以用如下指令:
#include“AT89S52.reg.h”
charin;
in=P15.2.2位的控制
AT89S52內(nèi)部有一些特殊功能寄存器,其中可以分別做位的控制,如P0、P1、P2、P3等。例如,將端口1位7設(shè)為高電位??梢允褂脜R編語言指令:
setbP1.7
將其設(shè)為低電位可以使用指令:
clrP1.7
將位取反可以使用指令:
cplP1.7
而在C51中則可以使用如下指令:#include"AT89S52io.h" /*一般I/O定義*/
#include"AT89S52lbit.h" /*位設(shè)定/取消宏定義*/
#include"AT89S52reg.h" /*8051寄存器定義*/
setbit(P1.7), /*設(shè)為高電位*/
clrbit(P1.7); /*設(shè)為低電位*/
cplbit(P1.7); /*位取反*/上述3指令在AT89S52bit.h定義中已用宏處理了。其內(nèi)容如下:
/*
*MacrostoallowdirectaccesstotheI/ObitsoftheAT89S52
*internalregistersfromDDSMICRO-C/51.
*REQUIRESTHEEXTERNALPREPROCESSOR(MCP)!
*
*Copyright1991-1994DaveDunfield
*Allrightsreserved.
*/
#definesetbit(bit)asm{/*MacrotosetaI/Obit*/
SETBbit
}#defineclrbit(bit)asm{/*MacrotoclearanI/Obit*/
CLRbit
}
Definecplbit(bit)asm{/*MacrotocomplemetanI/Obit*/
CPLbit
}5.2.3中斷子程序的設(shè)計(jì)
在有關(guān)中斷的頭文件AT89S52INT.H中,有關(guān)于中斷向量的執(zhí)行地址,如下所示:/*
*Common8025interruptvectoradresses
(Mustbedecimal)
*/
defineIEO3/*Externalinterrupt0*/
#defineTFO11/*Timer0rollover*/
defineIE119/*Externalinterrupt1*/
defineTF127/*Timer1rollover*/
#defineSER35/*SerialportRXorTX*/
#defineTF243/*Timer2rollover*/
defineEXF243/*Timer2externalflag*/若中斷程序?qū)憺槿缦碌某绦蚪Y(jié)構(gòu):
INTERRUPT(-TF0-)t0–isr()
{
程序內(nèi)容
}
【例5-11】以計(jì)時器0模式1做每隔5ms的計(jì)時中斷,即每隔5ms執(zhí)行一次中斷服務(wù)程序t0-isr()。
#include″AT89S52i0.h″
#include″AT89S52reg.h″
#include″AT89S52bit.h″
#include″AT89S52lint.h″
#dehneHI238/*5msrun1timeISR*/
#defineLO0
INTERRUPT(-TF0-)t0–isr()
{
TH0=HI;
TL0=LO;
}
main()
{
TMOD=0x01;
TH0=HI;
TL0=LO;
IE=0x82;
TCON=0x10;
}5.2.4內(nèi)存應(yīng)對式I/O
【例5-12】將外部RAM從2000H起的連續(xù)100字節(jié)全部填為“55H”。
程序如下:
#defineadd(char*)0x2000
test–ext-ram()
{
inti;
char*pt;
pt=add;
for(i=0;i<100;i++)
*pt++=0x55
}
【例5-13】從AT89S52單板上8255端口A送出數(shù)據(jù)“55H”。
程序如下:
#definepa(*(char*)0x8000)
definepb(*(char*)0x8001)
definepc(*(char*)0x8002)
definecr(*(char*)0x8003)
text-8255()
{
cr=0x80;
pa=0x55;
}5.2.5C51程序設(shè)計(jì)舉例
1.“求和”的C51程序設(shè)計(jì)
【例5-14】已知x=10,y=20,計(jì)算z=x+y的結(jié)果。
main() /*主函數(shù)名*/
{ /*主函數(shù)體開始*/
intx,y,z; /*主函數(shù)的內(nèi)部變量類型說明*/
x=10:y=20; /*變量賦值*/
z=x+y; /*計(jì)算z=x+y的值*/
} /*程序結(jié)束*/
2.求最大值的C51程序設(shè)計(jì)
【例5-15】求最大值。
#include〈stdio.h〉 /*預(yù)處理命令*/
include〈reg51.h〉
main() /*主函數(shù)名*/
{ /*主函數(shù)體開始*/
inta,A,c; /*主函數(shù)的內(nèi)部變量類型說明*/
intmax(intx,inty); /*功能函數(shù)max及其形式參數(shù)說明*/
SCON=0x52; /*8051單片機(jī)串行口初始化*/
TMOD=0x20;
TCON=0x0F3
scanf(″%d%d,&a,&A″)/*輸入變量a和A的值*/
c=max(a,A); /*調(diào)用max函數(shù)*/
printf(″max=%d″,c) /*輸出變量c的值*/
} /*主程序結(jié)束*/
intmax(intx,inty) /*定義max函數(shù),x、y為形式參數(shù)*/
{ /*max函數(shù)體開始*/
intz; /*max函數(shù)內(nèi)部變量類型說明*/
if(x>y)z=x /*計(jì)算最大值*/
elsez=y;
return(z);/*將計(jì)算得到的最大值z返回到調(diào)用處*/
} /*max函數(shù)結(jié)束*/
C51本身提供有“asm”語句,有以下兩種格式。
格式1:
asm″匯編語言指令″;
在此格式中,雙引號內(nèi)的所有指令將全部送到匯編語言編譯器。注意,在語句的最后要加上分號,作為語句的結(jié)束,如同一般的C語言程序。
格式2:
asm{
匯編語言指令
};5.3匯編語言與C語言的混合編程下面舉一個在C語言程序設(shè)計(jì)中加入?yún)R編語言語句的實(shí)例。
#include"AT89S52io.h"
#include"AT89S52reg.h"
#include"AT89S52bit"
/**/
text-asm0.
{
asm"nop";
asm"nop";
asm"clrP1.7"
}
/**/
text-asml()
{
asm{
NOP
NOP
CPLPI.7
}
}
/**/
main()
{
test-asm()
test-asm()
}5.3.1C51和A51接口所涉及的幾個主要問題
1.C51函數(shù)名的轉(zhuǎn)換及其命名規(guī)則
C51中函數(shù)名的轉(zhuǎn)換規(guī)則如表5-6所示。表5-6C51函數(shù)名的轉(zhuǎn)換規(guī)則
2.C51函數(shù)及其相關(guān)段的命名規(guī)則
依賴于所使用的存儲器模式,這些段的段名按表5-7所列規(guī)則命名,在相互調(diào)用時,匯編語言必須服從C51有關(guān)段名的命名規(guī)則。表5-7各種存儲器模式下函數(shù)相關(guān)段名的命名規(guī)則
3.C51函數(shù)的參數(shù)傳遞規(guī)則
表5-8是利用寄存器傳遞參數(shù)的規(guī)則。表5-8C51中的參數(shù)傳遞當(dāng)函數(shù)具有返回值時,也需傳遞參數(shù),這種返回值參數(shù)的傳遞均是通過CPU內(nèi)部寄存器完成,其傳遞規(guī)則如表5-9所示。表5-9函數(shù)返回使用的寄存器下面給出了一個C51程序及其編譯后的程序清單,從中可以更直觀地理解C51的函數(shù)名轉(zhuǎn)換規(guī)則、段命名規(guī)則及參數(shù)傳遞規(guī)則。
/*
文件名:ASM.c
功能:計(jì)算x/y
*/
#Include<reg51.h>
#deinfineucharunsignedchar
ucharfunc(ucharx,uchary); /*函數(shù)func原型聲明*/
voidmain(void) /*主函數(shù)*/
{
func(0x12,0x34); /*調(diào)用函數(shù)func*/
}
ucharfunc(ucharx,uchary) /*函數(shù)func*/
{
return(x/y) /*計(jì)算x/y并返回結(jié)果*/
}
;********************************
;文件名;ASM.SRC
;說明:此文件是ASM.C編譯后的匯編輸出文件(限于篇幅,有所省略)
;********************************
?PR?main?ASMSEGMENTCODE /*主函數(shù)main代碼段聲明*/
?PR?_func?ASMSEGMENTCOME /*函數(shù)func代碼段聲明*/
PUBLIC_func /*公開函數(shù)名以便可被其他模塊調(diào)用*/
PUBLICmain
RSEG?PR?main?ASM
main /*主函數(shù)代碼段起始*/
;{
;func(0x12,0x34);
MOVR7,#02H /*R7傳遞第一個char參數(shù)*/
MOVR5,#034H /*R5傳遞第二個char參數(shù)*/
LCALI_func /*調(diào)用函數(shù)func*/
;}
RET /*返回*/
;ucharfunc(ucharx,uchary)
RSEG?PR?_func?ASM_func; /*函數(shù)func代碼段起始*/
;{
;return(x/y);
MOVA,R7; /*計(jì)算x/y*/
MOVB,R5
DIVAB
MOVR7,A /*結(jié)果經(jīng)R7返回*/
;}
RET /*返回*/
END /*結(jié)束*/5.3.2C51程序中嵌入?yún)R編
(1)通過預(yù)編譯指令“#pragmaasm”和“#pragma.asm”在C語言代碼中插入?yún)R編語言代碼,如:
#include〈reg51.h〉
voidmain(void)
{
P2=1;
#pragmaasm//從此處插入?yún)R編代碼
MOVR7,#10
DEL:MOVR6,#20
DJNZR6,$
DJNZR7,DEL
#pragmaendasm//結(jié)束匯編代碼
P2=0;
}
(2)在project窗口中包含匯編代碼的C文件上單擊右鍵,選擇“Optionsfor…”選項(xiàng),單擊右邊的“GenerateassemblerSRCFile”和“AssembleSRCFile”,使復(fù)選框由灰色變成黑色狀態(tài),設(shè)置完成之后的對話框如圖5-2所示。
圖5-2SRC選項(xiàng)設(shè)置完成之后的對話框
(3)根據(jù)選擇的編譯模式,把相應(yīng)的庫文件(如Small模式時是Keil\C51\Lib\C51S.Lib)加入工程中,該文件必須作為工程的最后文件。如果沒有做這一步,編譯時會出現(xiàn)如下警告:
“UNRESOLVEDEXTERNALSYMBOL”。
(4)編譯,即可生成目標(biāo)代碼。5.3.3C51與匯編函數(shù)的相互調(diào)用
1)先用C語言編寫所有的代碼,包括需要匯編語言實(shí)現(xiàn)的部分。這里所選的例子就是前面討論的單片機(jī)模擬節(jié)能路燈的例子,延時子程序用匯編語言編寫,其余程序用C語言編寫。
則程序清單如下:/*
文件名:main.c
功能:模擬節(jié)能路燈的控制過程主程序
*/
#include〈AT89X51.H〉
SbitK1=P3^0;
SbitL1=P1^0;
externvoiddelay02s(void);//延時0.2s子程序
voidmain(void)
{
while(I)
{
if(K1==0){
L1=0; //點(diǎn)亮發(fā)光二極管
delay02s(); //調(diào)用延時子程序
L1=1; //熄滅發(fā)光二極管
}
else
{
L1=1; //熄滅發(fā)光二極管
}
}
}/*
文件名:delay.c
功能:延時0.2s
*/
delay02s(void)
{
unsignedchari,j,k;
for(i=20;i>0;i--)
for(j=20;j>0;j--)
for(k=248;k>0;k--);
}
(2)建立項(xiàng)目文件,將main.c和delay.c都包含進(jìn)項(xiàng)目。
(3)在project窗口中包含匯編代碼的.c文件上單擊右鍵,選擇“Optionsfor…”選項(xiàng),單擊右邊的“GenerateassemblerSRCFile”和“AssembleSRCFile”,使復(fù)選框由灰色變成黑色(有效)狀態(tài)。本例是在delay.c文件上進(jìn)行此操作。
(4)根據(jù)選擇的匯編模式,把相應(yīng)的庫文件(Small模式時,是Keil\C51\Lib\C51S.Lib)加入工程,該文件必須作為工程的最后文件。
(5)編譯這個項(xiàng)目后將會產(chǎn)生一個delay.SRC的文件,將這個文件名改為delay.A51。為了讓用戶更直觀地認(rèn)識C程序經(jīng)編譯器編譯之后的情況,下面給出了由編譯器產(chǎn)生的不做任何修改的delay.a51的清單:
;\delay.SRCgeneratedfrom:delay.c
;COMPILERINVOKEDBY:
;d:\keil\C51\BIN\C51.EXEdelay.cBROWSEDEBUGOBJECTEXTENDSRC(.\delay.SRC)
NAMEDELAY
?PR?delay02s?DELAYSEGMENTCODE
PUBLICdelay02S
;/*
;文件名:delay.c
;功能:延時0.2s
;*/
;voiddelay02s(void)
RSEG?PR?delay02s?DELAY
Delay02s
USING0
;SOURCELINE#5
;{
;SOURCELINE#6
;unsignedchari,j,k;
;for(i=20;i>0;i--)
;SOURCELINE#8;
;Variable′i?040′assignedtoRegister′R7′
MOVR7,#014H
?Cooo1:
;for(j=20;j>0;j--)
;SOURCELINE#9
;Variable′j?041′assignedtoRegister′R6′
MOVR6,#014H
?C0001:
;for(k=248;k>0;k--)
;SOURCELINE#10
;Variable′k?042′assignedtoregister′R5′
MOVR5,#0F8H
?C0007
DJNZR5,?C0007
?C0006
DJNZR6,?C0004
?C0003
DJNZR7,?C0001
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
- 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 人人文庫網(wǎng)僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負(fù)責(zé)。
- 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- GB/T 44929-2024空間高能天文科學(xué)觀測提案評估規(guī)范
- 2024政工程合同協(xié)議書:新能源項(xiàng)目合作協(xié)議3篇
- 2024建筑水電安裝系統(tǒng)檢測與維護(hù)合同
- 2024消防工程維保與消防系統(tǒng)智能化改造服務(wù)協(xié)議3篇
- 專業(yè)消防站施工建設(shè)協(xié)議樣本版A版
- 2024版專業(yè)土方外運(yùn)工程承包合同范本版B版
- 2024氣象站氣象儀器設(shè)備采購與維護(hù)服務(wù)協(xié)議3篇
- 2024靈活就業(yè)人員的勞動合同
- 專業(yè)液體化工品運(yùn)輸協(xié)議2024定制版版B版
- 個人房產(chǎn)轉(zhuǎn)讓協(xié)議范本2024版版B版
- 梅花鹿養(yǎng)殖基地產(chǎn)業(yè)化建設(shè)項(xiàng)目可行性研究報(bào)告(含財(cái)務(wù)表)
- 一年級帶拼音閱讀(全)
- 管理研究方法論for msci.students maxqda12入門指南
- 基于“產(chǎn)教結(jié)合”的電子商務(wù)專業(yè)實(shí)習(xí)實(shí)訓(xùn)教學(xué)評價體系
- TSEESA 010-2022 零碳園區(qū)創(chuàng)建與評價技術(shù)規(guī)范
- GB/T 3003-2017耐火纖維及制品
- GB/T 19867.5-2008電阻焊焊接工藝規(guī)程
- GB/T 18920-2020城市污水再生利用城市雜用水水質(zhì)
- 2023年市場部主管年終工作總結(jié)及明年工作計(jì)劃
- GB 17267-1998液化石油氣瓶充裝站安全技術(shù)條件
- 上期開特下期必開特規(guī)律
評論
0/150
提交評論