C单片机C语言编程.ppt
- 文档编号:18695494
- 上传时间:2023-09-17
- 格式:PPT
- 页数:55
- 大小:238KB
C单片机C语言编程.ppt
《C单片机C语言编程.ppt》由会员分享,可在线阅读,更多相关《C单片机C语言编程.ppt(55页珍藏版)》请在冰点文库上搜索。
第三章单片机C语言程序设计,3.1C语言与89C51单片机,3.1.1C语言的特点及程序结构,一C语言的特点1语言简洁、紧凑,使用方便、灵活。
2运算符丰富。
3数据结构丰富。
具有现代化语言的各种数据结构。
4可进行结构化程序设计。
5可以直接对计算机硬件进行操作。
6生成的目标代码质量高,程序执行效率高。
7可移植性好。
二C语言的程序结构,C语言程序采用函数结构,每个C语言程序由一个或多个函数组成,在这些函数中至少应包含一个主函数main(),也可以包含一个main()函数和若干个其它的功能函数。
不管main()函数放于何处,程序总是从main()函数开始执行,执行到main()函数结束则结束。
在main()函数中调用其它函数,其它函数也可以相互调用,但main()函数只能调用其它的功能函数,而不能被其它的函数所调用。
功能函数可以是C语言编译器提供的库函数,也可以是由用户定义的自定义函数。
在编制C程序时,程序的开始部分一般是预处理命令、函数说明和变量定义等。
3.1.2C语言与89C51单片机,用C语言编写89C51单片机程序与用汇编语言编写89C51单片机程序不一样。
用汇编语言编写89C51单片机程序必须要考虑其存储器结构,尤其必须考虑其片内数据存储器与特殊功能寄存器的使用以及按实际地址处理端口数据。
用C语言编写的89C51单片机应用程序,则不用像汇编语言那样须具体组织、分配存储器资源和处理端口数据,但在C语言编程中,对数据类型与变量的定义,必须要与单片机的存储结构相关联,否则编译器不能正确地映射定位。
C语言编写单片机应用程序时,需根据单片机存储结构及内部资源定义相应的数据类型和变量,而标准的C语言程序不需要考虑这些问题;C51包含的数据类型、变量存储模式、输入输出处理、函数等方面与标准的C语言有一定的区别。
其它的语法规则、程序结构及程序设计方法等与标准的C语言程序设计相同。
用C语言编写单片机应用程序与标准的C语言程序也有相应的区别:
现在支持89C51系列单片机的C语言编译器有很多种,如AmericanAutomation、Avocet、BSO/TASKING、DUNFIELDSHAREWARE、KEIL/Franklin等。
各种编译器的基本情况相同,但具体处理时有一定的区别,其中KEIL/Franklin以它的代码紧凑和使用方便等特点优于其它编译器,现在使用特别广泛。
本章主要以KEIL编译器介绍89C51单片机C语言程序设计。
3.1.3C51程序结构,C51的语法规定、程序结构及程序设计方法都与标准的C语言程序设计相同,但C51程序与标准的C程序在以下几个方面不一样:
(1)C51中定义的库函数和标准C语言定义的库函数不同。
标准的C语言定义的库函数是按通用微型计算机来定义的,而C51中的库函数是按89C51单片机相应情况来定义的;
(2)C51中的数据类型与标准C的数据类型也有一定的区别,在C51中还增加了几种针对89C51单片机特有的数据类型;,(3)C51变量的存储模式与标准C中变量的存储模式不一样,C51中变量的存储模式是与89C51单片机的存储器紧密相关;(4)C51与标准C的输入输出处理不一样,C51中的输入输出是通过89C51串行口来完成的,输入输出指令执行前必须要对串行口进行初始化;(5)C51与标准C在函数使用方面也有一定的区别,C51中有专门的中断函数。
3.2C51的数据类型,C51的数据类型分为基本数据类型和组合数据类型,情况与标准C中的数据类型基本相同,但其中char型与short型相同,float型与double型相同,另外,C51中还有专门针对于89C51单片机的特殊功能寄存器型和位类型。
特殊功能寄存器型这是C51扩充的数据类型,用于访问89C51单片机中的特殊功能寄存器数据,它分sfr和sfr16两种类型。
其中:
sfr为字节型特殊功能寄存器类型,占一个内存单元,利用它可以访问89C51内部的所有特殊功能寄存器;sfr16为双字节型特殊功能寄存器类型,占用两个字节单元,利用它可以访问89C51内部的所有两个字节的特殊功能寄存器。
在C51中对特殊功能寄存器的访问必须先用sfr或sfr16进行声明。
位类型这也是C51中扩充的数据类型,用于访问MCS-51单片机中的可寻址的位单元。
在C51中,支持两种位类型:
bit型和sbit型。
它们在内存中都只占一个二进制位,其值可以是“1”或“0”。
其中:
用bit定义的位变量在C51编译器编译时,在不同的时候位地址是可以变化的,而用sbit定义的位变量必须与89C51单片机的一个可以寻址位单元或可位寻址的字节单元中的某一位联系在一起,在C51编译器编译时,其对应的位地址是不可变化的。
KEILC51编译器能够识别的基本数据类型:
在C51语言程序中,有可能会出现在运算中数据类型不一致的情况。
C51允许任何标准数据类型的隐式转换,隐式转换的优先级顺序如下:
bitcharintlongfloatsignedunsigned也就是说,当char型与int型进行运算时,先自动对char型扩展为int型,然后与int型进行运算,运算结果为int型。
C51除了支持隐式类型转换外,还可以通过强制类型转换符“()”对数据类型进行人为的强制转换。
C5l编译器除了能支持以上这些基本数据类型之外,还能支持一些复杂的组合型数据类型,如数组类型、指针类型、结构类型、联合类型等这些复杂的数据类型,在后面将相继介绍。
3.3.1常量常量是指在程序执行过程中其值不能改变的量。
在C51中支持整型常量、浮点型常量、字符型常量和字符串型常量。
3.3C51的运算量,3.3.2变量变量是在程序运行过程中其值可以改变的量。
一个变量由两部分组成:
变量名和变量值。
在C51中,变量在使用前必须对变量进行定义,指出变量的数据类型和存储模式。
以便编译系统为它分配相应的存储单元。
定义的格式如下:
存储种类数据类型说明符存储器类型变量名1=初值,变量名2初值;,三存储种类存储种类是指变量在程序执行过程中的作用范围。
C51变量的存储种类有四种,分别是自动(auto)、外部(extern)、静态(static)和寄存器(register)。
1auto:
使用auto定义的变量称为自动变量,其作用范围在定义它的函数体或复合语句内部,当定义它的函数体或复合语句执行时,C51才为该变量分配内存空间,结束时占用的内存空间释放。
自动变量一般分配在内存的堆栈空间中。
定义变量时,如果省略存储种类,则该变量默认为自动(auto)变量。
2extern:
使用extern定义的变量称为外部变量。
在一个函数体内,要使用一个已在该函数体外或别的程序中定义过的外部变量时,该变量在该函数体内要用extern说明。
外部变量被定义后分配固定的内存空间,在程序整个执行时间内都有效,直到程序结束才释放。
3static:
使用static定义的变量称为静态变量。
它又分为内部静态变量和外部静态变量。
在函数体内部定义的静态变量为内部静态变量,它在对应的函数体内有效,一直存在,但在函数体外不可见,这样不仅使变量在定义它的函数体外被保护,还可以实现当离开函数时值不被改变。
外部静态变量上在函数外部定义的静态变量。
它在程序中一直存在,但在定义的范围之外是不可见的。
如在多文件或多模块处理中,外部静态变量只在文件内部或模块内部有效。
4register:
使用register定义的变量称为寄存器变量。
它定义的变量存放在CPU内部的寄存器中,处理速度快,但数目少。
C51编译器编译时能自动识别程序中使用频率最高的变量,并自动将其作为寄存器变量,用户可以无需专门声明。
四存储器类型存储器类型是用于指明变量所处的单片机的存储器区域情况。
存储器类型与存储种类完全不同。
C51编译器能识别的存储器类型有以下几种,见表所示。
定义变量时也可以省“存储器类型”,省时C51编译器将按编译模式默认存储器类型,具体编译模式的情况在后面介绍。
【例】变量定义存储种类和存储器类型相关情况。
chardatavarl;/*在片内RAM低128B定义用直接寻址方式访问的字符型变量var1*/intidatavar2;/*在片内RAM256B定义用间接寻址方式访问的整型变量var2*/autounsignedlongdatavar3;/*在片内RAM128B定义用直接寻址方式访问的自动无符号长整型变量var3*/externfloatxdatavar4;/*在片外RAM64KB空间定义用间接寻址方式访问的外部实型变量var4*/intcodevar5;/*在ROM空间定义整型变量var5*/unsigncharbdatavar6;/*在片内RAM位寻址区20H2FH单元定义可字节处理和位处理的无符号字符型变量var6*/,五特殊功能寄存器变量89C51系列单片机片内有许多特殊功能寄存器,通过这些特殊功能寄存器可以控制89C51系列单片机的定时器、计数器、串口、I/O及其它功能部件,每一个特殊功能寄存器在片内RAM中都对应于一个字节单元或两个字节单元。
在C51中,允许用户对这些特殊功能寄存器进行访问,访问时须通过sfr或sfr16类型说明符进行定义,定义时须指明它们所对应的片内RAM单元的地址。
格式如下:
sfr或sfr16特殊功能寄存器名=地址;sfr用于对89C51单片机中单字节的特殊功能寄存器进行定义,sfr16用于对双字节特殊功能寄存器进行定义。
特殊功能寄存器名一般用大写字母表示。
地址一般用直接地址形式,具体特殊功能寄存器地址见前面内容。
【例3-3】特殊功能寄存器的定义。
sfrPSW=0xd0;sfrSCON=0x98;sfrTMOD=0x89;sfrP1=0x90;sfr16DPTR=0x82;sfr16T1=0X8A;,六位变量在C51中,允许用户通过位类型符定义位变量。
位类型符有两个:
bit和sbit。
可以定义两种位变量。
bit位类型符用于定义一般的可位处理位变量。
它的格式如下:
bit位变量名;在格式中可以加上各种修饰,但注意存储器类型只能是bdata、data、idata。
只能是片内RAM的可位寻址区,严格来说只能是bdata。
【例】bit型变量的定义。
bitdataa1;/*正确*/bitbdataa2;/*正确*/bitpdataa3;/*错误*/bitxdataa4;/*错误*/,sbit位类型符用于定义在可位寻址字节或特殊功能寄存器中的位,定义时须指明其位地址,可以是位直接地址,可以是可位寻址变量带位号,也可以是特殊功能寄存器名带位号。
格式如下:
sbit位变量名=位地址;如位地址为位直接地址,其取值范围为0x000xff;如位地址是可位寻址变量带位号或特殊功能寄存器名带位号,则在它前面须对可位寻址变量或特殊功能寄存器进行定义。
字节地址与位号之间、特殊功能寄存器与位号之间一般用“”作间隔。
【例】sbit型变量的定义:
sbitOV=0xd2;sbitCY=oxd7;unsignedcharbdataflag;sbitflag0=flag0;sfrP1=0x90;sbitP1_0=P10;sbitP1_1=P11;sbitP1_2=P12;sbitP1_3=P13;sbitP1_4=P14;sbitP1_5=P15;sbitP1_6=P16;sbitP1_7=P17;,在C51中,为了用户处理方便,C51编译器把89C51单片机的常用的特殊功能寄存器和特殊位进行了定义,放在一个“reg51.h”或“reg52.h”的头文件中,当用户要使用时,只须要在使用之前用一条预处理命令#include把这个头文件包含到程序中,然后就可使用殊功能寄存器名和特殊位名称。
4.3.3存储模式,C51编译器支持三种存储模式:
SMALL模式、COMPACT模式和LARGE模式。
不同的存储模式对变量默认的存储器类型不一样。
(1)SMALL模式。
SMALL模式称为小编译模式,在SMALL模式下,编译时,函数参数和变量被默认在片内RAM中,存储器类型为data。
(2)COMPACT模式。
COMPACT模式称为紧凑编译模式,在COMPACT模式下,编译时,函数参数和变量被默认在片外RAM的低256字节空间,存储器类型为pdata。
(3)LARGE模式。
LARGE模式称为大编译模式,在LARGE模式下,编译时函数参数和变量被默认在片外RAM的64K字节空间,存储器类型为xdata。
在程序中变量的存储模式的指定通过#pragma预处理命令来实现。
函数的存储模式可通过在函数定义时后面带存储模式说明。
如果没有指定,则系统都隐含为SMALL模式。
【例】变量的存储模式。
#pragmasmall/*变量的存储模式为SMALL*/chark1;intxdatam1;#pragmacompact/*变量的存储模式为compact*/chark2;intxdatam2;intfunc1(intx1,inty1)large/*函数的存储模式为LARGE*/return(x1+y1);intfunc2(intx2,inty2)/*函数的存储模式隐含为SMALL*/return(x2-y2);,程序编译时,k1变量存储器类型为data,k2变量存储器类型为pdata,而m1和m2由于定义时带了存储器类型xdata,因而它们为xdata型;函数func1的形参x1和y1的存储器类型为xdata型,而函数func2由于没有指明存储模式,隐含为SMALL模式,形参x2和y2的存储器类型为data。
3.3.4绝对地址的访问,一使用C51运行库中预定义宏C51编译器提供了一组宏定义来对51系列单片机的code、data、pdata和xdata空间进行绝对寻址。
规定只能以无符号数方式访问,定义了8个宏定义,其函数原型如下:
#defineCBYTE(unsignedcharvolatile*)0x50000L)#defineDBYTE(unsignedcharvolatile*)0x40000L)#definePBYTE(unsignedcharvolatile*)0x30000L)#defineXBYTE(unsignedcharvolatile*)0x20000L)#defineCWORD(unsignedintvolatile*)0x50000L)#defineDWORD(unsignedintvolatile*)0x40000L)#definePWORD(unsignedintvolatile*)0x30000L)#defineXWORD(unsignedintvolatile*)0x20000L)这些函数原型放在absacc.h文件中。
使用时须用预处理命令把该头文件包含到文件中,形式为:
#include。
其中:
CBYTE以字节形式对code区寻址,DBYTE以字节形式对data区寻址,PBYTE以字节形式对pdata区寻址,XBYTE以字节形式对xdata区寻址,CWORD以字形式对code区寻址,DWORD以字形式对data区寻址,PWORD以字形式对pdata区寻址,XWORD以字形式对xdata区寻址。
【例】绝对地址对存储单元的访问#include/*将绝对地址头文件包含在文件中*/#include/*将寄存器头文件包含在文件中*/#defineucharunsignedchar/*定义符号uchar为数据类型符unsignedchar*/#defineuintunsignedint/*定义符号uint为数据类型符unsignedint*/voidmain(void)ucharvar1;uintvar2;var1=XBYTE0x0005;/*XBYTE0x0005访问片外RAM的0005字节单元*/var2=XWORD0x0002;/*XWORD0x0002访问片外RAM的000字单元*/.while
(1);,在上面程序中,其中XBYTE0x0005就是以绝对地址方式访问的片外RAM0005字节单元;XWORD0x0002就是以绝对地址方式访问的片外RAM0002字单元。
【例】通过指针实现绝对地址的访问。
#defineucharunsignedchar/*定义符号uchar为数据类型符unsignedchar*/#defineuintunsignedint/*定义符号uint为数据类型符unsignedint*/voidfunc(void)uchardatavar1;ucharpdata*dp1;/*定义一个指向pdata区的指针dp1*/uintxdata*dp2;/*定义一个指向xdata区的指针dp2*/uchardata*dp3;/*定义一个指向data区的指针dp3*/dp1=0x30;/*dp1指针赋值,指向pdata区的30H单元*/dp2=0x1000;/*dp2指针赋值,指向xdata区的1000H单元*/*dp1=0xff;/*将数据0xff送到片外RAM30H单元*/*dp2=0x1234;/*将数据0x1234送到片外RAM1000H单元*/dp3=/*给变量var1赋值0x20*/,二通过指针访问采用指针的方法,可以实现在C51程序中对任意指定的存储器单元进行访问。
三使用C51扩展关键字_at_使用_at_对指定的存储器空间的绝对地址进行访问,一般格式如下:
存储器类型数据类型说明符变量名_at_地址常数;其中,存储器类型为data、bdata、idata、pdata等C51能识别的数据类型,如省略则按存储模式规定的默认存储器类型确定变量的存储器区域;数据类型为C51支持的数据类型。
地址常数用于指定变量的绝对地址,必须位于有效的存储器空间之内;使用_at_定义的变量必须为全局变量。
【例】通过_at_实现绝对地址的访问。
#defineucharunsignedchar/*定义符号uchar为数据类型符unsignedchar*/#defineuintunsignedint/*定义符号uint为数据类型符unsignedint*/voidmain(void)dataucharx1_at_0x40;/*在data区中定义字节变量x1,它的地址为40H*/xdatauintx2_at_0x2000;/*在xdata区中定义字变量x2,它的地址为2000H*/x1=0xff;x2=0x1234;.while
(1);,3.4C51的运算符及表达式,3.4.1赋值运算3.4.2算术运算3.4.3逻辑运算3.4.4位运算3.4.5复合赋值运算3.4.6逗号运算3.4.7条件运算3.4.8指针与地址运算,3.5表达式语句及复合语句,3.5.1表达式语句,在表达式的后边加一个分号“;”就构成了表达式语句,如:
a=+b*9;x=8;y=7;+k;可以一行放一个表达式形成表达式语句,也可以一行放多个表达式形成表达式语句,这时每个表达式后面都必须带“;”号,另外,还可以仅由个分号“;”占一行形成一个表达式语句,这种语句称为空语句。
空语句在程序设计中通常用于两种情况:
(1)在程序中为有关语句提供标号,用以标记程序执行的位置。
例如采用下面的语句可以构成一个循环。
repeat:
;gotorepeat;
(2)在用while语句构成的循环语句后面加一个分号,形成一个不执行其它操作的空循环体。
这种结构通常用于对某位进行判断,当不满足条件则等待,满足条件则执行。
【例】下面这段子程序用于读取8051单片机的串行口的数据,当没有接收到则等待,当接收到,接收数据后返回,返回值为接收的数据。
#includechargetchar()charc;while(!
RI);/当接收中断标志位RI为0则等待,当接收中断标志位为1则;等待结束c=SBUF;RI=0;return(c);,3.5.2复合语句,复合语句是由若干条语句组合而成的一种语句,在C51中,用一个大括号“”将若干条语句括在一起就形成了一个复合语句,复合语句最后不需要以分号“;”结束,但它内部的各条语句仍需以分号“;”结束。
复合语句的一般形式为:
局部变量定义;语句l;语句2;,复合语句在执行时,其中的各条单语句按顺序依次执行,整个复合语句在语法上等价于一条单语句,因此在C51中可以将复合语句视为一条单语句。
通常复合语句出现在函数中,实际上,函数的执行部分(即函数体)就是一个复合语句;复合语句中的单语句一般是可执行语句,此外还可以是变量的定义语句(说明变量的数据类型)。
在复合语句内部语句所定义的变量,称为该复合语句中的局部变量,它仅在当前这个复合语句中有效。
利用复合语句将多条单语句组合在起,以及在复合语句中进行局部变量定义是C51语言的一个重要特征。
3.6C51的输入输出,在C51语言中,它本身不提供输入和输出语句,输入和输出操作是由函数来实现的。
在C51的标准函数库中提供了一个名为“stdio.h”的一般I/O函数库,它当中定义了C51中的输入和输出函数。
当对输入和输出函数使用时,须先用预处理命令“#include”将该函数库包含到文件中。
在C51的一般I/O函数库中定义的I/O函数都是通过串行接口实现,在使用I/O函数之前,应先对MCS-51单片机的串行接口进行初始化。
选择串口工作于方式2(8位自动重载方式),波特率由定时器/计数器1溢出率决定。
例如,设系统时钟为12MHZ,波特率为2400,则初始化程序如下:
SCON=0x52;TMOD=0X20;TH1=0xf3;TR1=1;,3.6.1格式输出函数printf(),printf()函数的的作用是通过串行接口输出若干任意类型的数据,它的格式如下:
printf(格式控制,输出参数表)格式控制是用双引号括起来的字符串,也称转换控制字符串,它包括三种信息:
格式说明符、普通字符和转义字符。
(1)格式说明符,由“%”和格式字符组成,它的作用是用于指明输出的数据的格式输出,如%d、%f等,它们的具体情况见下表。
(2)普通字符,这些字符按原样输出,用来输出某些提示信息。
(3)转义字符,就是前面介绍的转义字符(下表),用来输出特定的控制符,如输出转义字符n就是使输出换一行。
输出参数表是需要输出的一组数据,可以是表达式。
3.6.2格式输入函数scanf(),scanf()函数的作用是通过串行接口实现数据输入,它的使用方法与printf()类似,scanf()的格式如下:
scanf(格式控制,地址列表)格式控制与printf()函数的情况类似,也是用双引号括起来的一些字符,可以包括以下三种信息:
空白字符、普通字符和格式说明。
(1)空白字符,包含空格、制表符、换行符等,这些字符在输出时被忽略。
(2)普通字符,除了以百分号“%”开头的格式说明符而外的所有非空白字符,在输入时要求原样输入。
(3)格式说明,由百分号“%”和格式说明符组成,用于指明输入数据的格式,它的基本情况与printf()相同,具体情况见表4-5。
地址列表是由若干个地址组成,它可以是指针变量、取地址运算符“&”加变量(变量的地址)或字符串名(表示字符串的首地址)。
【例12】使用格式输入输出函数的例子#include/包含特殊功能寄存器库#include/包含I/O函数库voidmain(void)/主函数intx,y;/定义整型变量x和ySCON=0x52;/串口初始化TMOD=0x20;TH1=0XF3;TR1=1;printf(“inputx,y:
n”);/输出提示信息scanf(“%d%d”,/结束,3.7C51程序基本结构与相关语句,3.7.1C51的基本结构,一、顺序结构二、选择结构三、循环结构,3.7.2C5
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- 单片机 语言 编程