单片机 C 语言入门教程
51单片机 C 语言入门教程
学习单片机实在不是件易事,一来要购买高价格的编程器,仿真器,二来要学习编程语言,还有众多种类的单片机选择真是件让人头痛的事。在众多单片机中 51 架构的芯片风行很久,学习资料也相对很多,是初学的较好的选择之一。51 的编程语言常用的有二种,一种是汇编语言,一种是 C 语言。汇编语言的机器代码生成效率很高但可读性却并不强,复杂一点的程序就更是难读懂,而 C 语言在大多数情况下其机器代码生成效率和汇编语言相当,但可读性和可移植性却远远超过汇编语言,而且 C 语言还可以嵌入汇编来解决高时效性的代码编写问题。对于开发周期来说,中大型的软件编写用 C 语言的开发周期通常要小于汇编语言很多。综合以上 C 语言的优点,我在学习时选择了 C 语言。以后的教程也只是我在学习过程中的一些学习笔记和随笔,在这里加以整理和修改,希望和大家一起分享,一起交流,一起学习,一起进步。
第一课 建立您的第一个 C 项目
使用 C 语言肯定要使用到 C 编译器,以便把写好的 C 程序编译为机器码,这样单片机才能执行编写好的程序。KEIL uVISION2 是众多单片机应用开发软件中优秀的软件之一,它支持众多不同公司的 MCS51 架构的芯片,它集编辑,编译,仿真等于一体,同时还支持,PLM,汇编和 C 语言的程序设计,它的界面和常用的微软 VC++的界面相似,界面友好,易学易用,在调试程序,软件仿真方面也有很强大的功能。因此很多开发 51 应用的工程师或普通的单片机爱好者,都对它十分喜欢。
以上简单介绍了 KEIL51 软件,要使用 KEIL51 软件,必需先要安装它。KEIL51 是一个商业的软件,对于我们这些普通爱好者可以到 KEIL 中国代理周立功公司的网站上下载一份能编译 2K 的 DEMO 版软件,基本可以满足一般的个人学习和小型应用的开发。(安装的方法和普通软件相当这里就不做介绍了)
安装好后,您是不是迫不及待的想建立自己的第一个 C 程序项目呢?下面就让我们一起来建立一个小程序项目吧。或许您手中还没有一块实验板,甚至没有一块单片机,不过没有关系我们可以通过 KEIL 软件仿真看到程序运行的结果。
首先当然是运行 KEIL51 软件。怎么打开?噢,天!那您要从头学电脑了。呵呵,开个玩笑,这个问题我想读者们也不会提的了:P。运行几秒后,出现如图 1-1 的屏幕。
图 1-1 启动时的屏幕
接着按下面的步骤建立您的第一个项目:
(1)点击 Project 菜单,选择弹出的下拉式菜单中的 New Project,如图 1-2。接着弹出一个标准 Windows 文件对话窗口,如图 1-3,这个东东想必大家是见了 N 次的了,用法技巧也不是这里要说的,以后的章节中出现类似情况将不再说明。在"文件名"中输入您的第一个 C 程序项目名称,这里我们用"test",这是笔者惯用的名称,大家不必照搬就是了,只要符合 Windows 文件规则的文件名都行。"保存"后的文件扩展名为 uv2,这是 KEIL uVision2 项目文件扩展名,以后我们可以直接点击此文件以打开先前做的项目。
图 1-2 New Project 菜单
·
图 1-3文件窗口
(2)选择所要的单片机,这里我们选择常用的 Ateml 公司的 AT89C51。此时屏幕如图1-4 所示。AT89C51 有什么功能、特点呢?不用急,看图中右边有简单的介绍,稍后的章节会作较详细的介绍。完成上面步骤后,我们就可以进行程序的编写了。
(3)首先我们要在项目中创建新的程序文件或加入旧程序文件。如果您没有现成的程序,那么就要新建一个程序文件。在 KEIL 中有一些程序的 Demo,在这里我们还是以一个 C 程序为例介绍如何新建一个 C 程序和如何加到您的第一个项目中吧。点击图 1-5 中 1 的新建文件的快捷按钮,在 2 中出现一个新的文字编辑窗口,这个操作也可以通过菜单 File -New 或快捷键 Ctrl+N 来实现。好了,现在可以编写程序了,光标已出现在文本编辑窗口中,等待我们的输入了。第一程序嘛,写个简单明了的吧。下面是经典的一段程序,呵,如单片机 C 语言入门教程
如果您看过别的程序书也许也有类似的程序:
#include <AT89X51.H>
#include <stdio.h>
void main(void)
{
SCON = 0x50; //串口方式 1,允许接收
TMOD = 0x20; //定时器 1 定时方式 2
TCON = 0x40; //设定时器 1 开始计数
TH1 = 0xE8; //11.0592MHz 1200 波特率
TL1 = 0xE8;
TI = 1;
TR1 = 1; //启动定时器
while(1)
{
printf ("Hello World!\n"); //显示 Hello World
}
}
图 1-4 选取芯片
图 1-5 新建程序文件
这段程序的功能是不断从串口输出"Hello World!"字符,我们先不管程序的语法和意思吧,先看看如何把它加入到项目中和如何编译试运行。
(4)点击图 1-5 中的 3 保存新建的程序,也可以用菜单 File-Save 或快捷键 Ctrl+S 进行保存。因是新文件所以保存时会弹出类似图 1-3 的文件操作窗口,我们把第一个程序命名为 test1.c,保存在项目所在的目录中,这时您会发现程序单词有了不同的颜色,说明 KEIL 的 C 语法检查生效了。如图 1-6 鼠标在屏幕左边的 Source Group1 文件夹图标上右击弹出菜单,在这里可以做在项目中增加减少文件等操作。我们选"Add File to Group 'Source Group 1'"弹出文件窗口,选择刚刚保存的文件,按 ADD 按钮,关闭文件窗,程序文件已加到项目中了。这时在 Source Group1 文件夹图标左边出现了一个小+号说明,文件组中有了文件,点击它可以展开查看。
图 1-6 把文件加入到项目文件组中
·
单片机 C 语言入门教程
(5)C 程序文件已被我们加到了项目中了,下面就剩下编译运行了。这个项目我们只是用做学习新建程序项目和编译运行仿真的基本方法,所以使用软件默认的编译设置,它不会生成用于芯片烧写的 HEX 文件,如何设置生成 HEX 文件就请看下面的第三课。我们先来看图 1-7 吧,图中 1、2、3 都是编译按钮,不同是 1 是用于编译单个文件。2 是编译当前项目,如果先前编译过一次之后文件没有做动编辑改动,这时再点击是不会再次重新编译的。3 是重新编译,每点击一次均会再次编译链接一次,不管程序是否有改动。在 3 右边的是停止编译按钮,只有点击了前三个中的任一个,停止按钮才会生效。5 是菜单中的它们,我个人就不习惯用它了。嘿嘿,这个项目只有一个文件,您按 123 中的一个都可以编译。按了?好快哦,呵呵。在 4 中可以看到编译的错误信息和使用的系统资源情况等,以后我们要查错就靠它了。6 是有一个小放大镜的按钮,这就是开启\关闭调试模式的按钮,它也存在于菜单 Debug-Start\Stop Debug Session,快捷键为 Ctrl+F5。
图 1-7 编译程序
(6)进入调试模式,软件窗口样式大致如图 1-8 所示。图中 1 为运行,当程序处于停止状态时才有效,2 为停止,程序处于运行状态时才有效。3 是复位,模拟芯片的复位,程序回到最开头处执行。按 4 我们可以打开 5 中的串行调试窗口,这个窗口我们可以看到从 51 芯片的串行口输入输出的字符,这里的第一个项目也正是在这里看运行结果。这些在菜单中也有,这里不再一一介绍大家不妨找找看,其它的功能也会在后面的课程中慢慢介绍。首先
· 4 打开串行调试窗口,再按运行键,这时就可以看到串行调试窗口中不断的打印"Hello World!"。呵呵,是不是不难呀?这样就完成了您的第一个 C 项目。最后我们要停止程序运行回到文件编辑模式中,就要先按停止按钮再按开启\关闭调试模式按钮。然后我们就可以进行关闭 KEIL 等相关操作了。
到此为止,第一课已经完结了,初步学习了一些 KEIL uVision2 的项目文件创建、编译、运行和软件仿真的基本操作方法。其中一直有提到一些功能的快捷键的使用,的确在实际的开发应用中快捷键的运用可以大大提高工作的效率,建议大家多多使用,还有就是对这里所讲的操作方法举一反三用于类似的操作中。
图 1-8 调试运行程序
第二课 初步认识 51 芯片
上一课我们的第一个项目完成了,可能有懂 C 语言的朋友会说,"这和 PC 机上的 C 语言没有多大的区别呀"。的确没有太大的区别,C 语言只是一种程序语言的统称,针对不同的处理器相关的 C 语言都会有一些细节的改变。编写 PC 机的 C 程序时,如要对硬件编程您就必须对硬件要有一定的认识,51 单片机编程就更是如此,因它的开发应用是不可与硬件脱节的,所以我们先要来初步认识一下 51 苾片的结构和引脚功能。MSC51 架构的芯片种类很多,具体特点和功能不尽相同(在以后编写的附录中会加入常用的一些 51 芯片的资料列表),在此后的教程中就以 Atmel 公司的 AT89C51 和 AT89C2051 为中心对象来进行学习,两者是 AT89 系列的典型代表,在爱好者中使用相当的多,应用资料很多,价格便宜,是初学 51 的首选芯片。嘿嘿,口水多多有点卖广告之嫌了。:P
图 2-1AT89C51 和 AT89C2051 引脚功能图
图 2-1 中是 AT89C51 和 AT89C2051 的引脚功能图。而表 2-1 中则是它们的主要性能表。以上可以看出它们是大体相同的,由于 AT89C2051 的 IO 线很少,导致它无法外加 RAM 和程序 ROM,片内 Flash 存储器也少,但它的体积比 AT89C51 小很多,以后大家可根据实际需要来选用。它们各有其特点但其核心是一样的,下面就来看看 AT89C51 的引脚具体功能。
1.电源引脚
Vcc40电源端
GND20接地端
*工作电压为 5V,另有 AT89LV51 工作电压则是 2.7-6V, 引脚功能一样。
2.外接晶体引脚
图 2-2外接晶体引脚
XTAL119
XTAL218
XTAL1 是片内振荡器的反相放大器输入端,XTAL2 则是输出端,使用外部振荡器时,外部振荡信号应直接加到 XTAL1,而 XTAL2 悬空。内部方式时,时钟发生器对振荡脉冲二分频,如晶振为 12MHz,时钟频率就为 6MHz。晶振的频率可以在 1MHz-24MHz 内选择。电容取 30PF 左右。
*型号同样为 AT89C51 的芯片,在其后面还有频率编号,有 12,16,20,24MHz 可选。大家在购买和选用时要注意了。如 AT89C51 24PC 就是最高振荡频率为 24MHz,40P6 封装的普通商用芯片。
3.复位RST9
在振荡器运行时,有两个机器周期(24 个振荡周期)以上的高电平出现在此引腿时,将使单片机复位,只要这个脚保持高电平,51 芯片便循环复位。复位后 P0-P3 口均置 1 引脚表现为高电平,程序计数器和特殊功能寄存器 SFR 全部清零。当复位脚由高电平变为低电平时,芯片为 ROM 的 00H 处开始运行程序。常用的复位电路如图 2-
3 所示。
*复位操作不会对内部 RAM 有所影响。
图 2-3常用复位电路
4.输入输出引脚
(1) P0 端口[P0.0-P0.7] P0 是一个 8 位漏极开路型双向 I/O 端口,端口置 1(对端口写 1)时作高阻抗输入端。作为输出口时能驱动 8 个 TTL。
对内部 Flash 程序存储器编程时,接收指令字节;校验程序时输出指令字节,要求外接上拉电阻。
在访问外部程序和外部数据存储器时,P0 口是分时转换的地址(低 8 位)/数据总线,访问期间内部的上拉电阻起作用。
(2) P1 端口[P1.0-P1.7] P1 是一个带有内部上拉电阻的 8 位双向 I/0 端口。输出时可驱动 4 个 TTL。端口置 1 时,内部上拉电阻将端口拉到高电平,作输入用。对内部 Flash 程序存储器编程时,接收低 8 位地址信息。
(3) P2 端口[P2.0-P2.7] P2 是一个带有内部上拉电阻的 8 位双向 I/0 端口。输出时可驱动 4 个 TTL。端口置 1 时,内部上拉电阻将端口拉到高电平,作输入用。对内部 Flash 程序存储器编程时,接收高 8 位地址和控制信息。
在访问外部程序和 16 位外部数据存储器时,P2 口送出高 8 位地址。而在访问 8 位地址的外部数据存储器时其引脚上的内容在此期间不会改变。
(4) P3 端口[P3.0-P3.7] P2 是一个带有内部上拉电阻的 8 位双向 I/0 端口。输出时可驱动 4 个 TTL。端口置 1 时,内部上拉电阻将端口拉到高电平,作输入用。对内部 Flash 程序存储器编程时,接控制信息。除此之外 P3 端口还用于一些专门功能,具体请看 表 2-2.。
*P1-3 端口在做输入使用时,因内部有上接电阻,被外部拉低的引脚会输出一定的电流。
呼!一口气说了那么多,停一下吧。嗯,什么?什么叫上拉电阻?上拉电阻简单来说就是把电平拉高,通常用 4.7-10K 的电阻接到 Vcc 电源,下拉电阻则是把电平拉低,电阻接
· GND 地线上。具体说明也不是这里要讨论的,接下来还是接着看其它的引脚功能吧。5.其它的控制或复用引脚
(1) ALE/PROG 30 访问外部存储器时,ALE(地址锁存允许)的输出用于锁存地址的低位字节。即使不访问外部存储器,ALE 端仍以不变的频率输出脉冲信号(此频率是振荡器频率的 1/6)。在访问外部数据存储器时,出现一个 ALE 脉冲。对 Flash 存储器编程时,这个引脚用于输入编程脉冲 PROG
(2) PSEN 29 该引是外部程序存储器的选通信号输出端。当 AT89C51 由外部程序存储器取指令或常数时,每个机器周期输出 2 个脉冲即两次有效。但访问外部数据存储器时,将不会有脉冲输出。
(3) EA/Vpp 31 外部访问允许端。当该引脚访问外部程序存储器时,应输入低电平。要使 AT89C51 只访问外部程序存储器(地址为 0000H-FFFFH),这时该引脚必须保持低电平。对 Flash 存储器编程时,用于施加 Vpp 编程电压。Vpp 电压有两种,类似芯片最大频率值要根据附加的编号或芯片内的特征字决定。具体如表2-3 所列。
看到这您对 AT89C51 引脚的功能应该有了一定的了解了,引脚在编程和校验时的时序我们在这里就不做详细的探讨,通常情况下我们也没有必要去撑握它,除非您想自己开发编程器。下来的课程我们要开始以一些简单的实例来讲述 C 程序的语法和编写方法技巧,中间穿插相关的硬件知识如串口,中断的用法等等。
第三课 生成 HEX 文件和最小化系统
在开始C语言的主要内容时,我们先来看看如何用KEIL uVISION2来编译生成用于烧写芯片的HEX文件。HEX文件格式是Intel公司提出的按地址排列的数据信息,数据宽度为字节,所有数据使用16进制数字表示, 常用来保存单片机或其他处理器的目标程序代码。它保存物理程序存储区中的目标代码映象。一般的编程器都支持这种格式。我们先来打开第一课做的第一项目,打开它的所在目录,找到test.Uv2的文件就可以打开先前的项目了。然后右击图3-1中的1项目文件夹,弹出项目功能菜单,选Options for Target'Target1',弹出项目选项
设置窗口,同样先选中项目文件夹图标,这时在Project菜单中也有一样的菜单可选。打开项目选项窗口,转到Output选项页图3-2所示,图中1是选择编译输出的路径,2是设置编译输出生成的文件名,3则是决定是否要创建HEX文件,选中它就可以输出HEX文件到指定的路径中。选好了?好,我们再将它重新编译一次,很快在编译信息窗口中就显示HEX文件创建到指定的路径中了,如图3-3。这样我们就可用自己的编程器所附带的软件去读取并烧到芯片了,再用实验板看结果,至于编程器或仿真器品种繁多具体方法就看它的说明书了,这里也不做讨论。
(技巧:一、在图3-1中的1里的项目文件树形目录中,先选中对象,再单击它就可对它进行重命名操作,双击文件图标便可打开文件。二、在Project下拉菜单的最下方有最近编辑过的项目路径保存,这里可以快速打开最近在编辑的项目。)
图3-1项目功能菜单
。
图3-2项目选项窗口
图 3-3编译信息窗口
或许您已把编译好的文件烧到了芯片上,如果您购买或自制了带串口输出元件的学习实验板,那您就可以把串口和 PC 机串口相联用串口调试软件或 Windows 的超级终端,将其波特率设为 1200,就可以看到不停输出的"Hello World!"字样。也许您还没有实验板,那这里先说说 AT89C51 的最小化系统,再以一实例程序验证最小化系统是否在运行,这个最小化系统也易于自制用于实验。图 3-4 便是 AT89C51 的最小化系统,不过为了让我们可以看出它是在运行的,我加了一个电阻和一个 LED,用以显示它的状态,晶振可以根据自己的情况使用,一般实验板上是用 11.0592MHz 或 12MHz,使用前者的好外是可以产生标准的串口波特率,后者则一个机器周期为 1 微秒,便于做精确定时。在自己做实验里,注意的是 VCC 是+5V 的,不能高于此值,否则将损坏单片机,太低则不能正常工作。在 31 脚要接高电平,这样我们才能执行片内的程序,如接低电平则使用片外的程序存储器。下面,我们建一个新的项目名为 OneLED 来验证最小化系统是否可以工作(所有的例程都可在我的主页下面下载到,网址: http://cdle.yeah.net 或 http://cdle.126.com )。程序如下:
#include <AT89X51.h> //预处理命令
void main(void) //主函数名
{
//这是第一种注释方式
unsigned int a; //定义变量 a 为 int 类型
/*
这是第二种注释方式
*/
do{ //do while 组成循环
for (a=0; a<50000; a++); //这是一个循环
P1_0 = 0; //设 P1.0 口为低电平,点亮 LED
for (a=0; a<50000; a++); //这是一个循环
P1_0 = 1; //设 P1.0 口为高电平,熄灭 LED
}
while(1);
}
图 3-4AT89C51 最小化系统
这里先讲讲 KEIL C 编译器所支持的注释语句。一种是以"//"符号开始的语句,符号之后的语句都被视为注释,直到有回车换行。另一种是在"/*"和"*/"符号之内的为注释。注释不会被 C 编译器所编译。一个 C 应用程序中应有一个 main 主函数,main 函数可以调用别的功能函数,但其它功能函数不允许调用 main 函数。不论 main 函数放在程序中的那个位置,总是先被执行。用上面学到的知识编译写好的 OneLED 程序,并把它烧到刚做好的最小化系统中。上电,刚开始时 LED 是不亮的(因为上电复位后所有的 IO 口都置 1 引脚为高电平),然后延时一段时间(for (a=0; a<50000; a++)这句在运行),LED 亮,再延时,LED熄灭,然后交替亮、灭。第一个真正的小应用就做完,呵呵,先不要管它是否实用哦。如果没有这样的效果那么您就要认真检查一下电路或编译烧写的步骤了。
第四课 数据类型
先来简单说说 C 语言的标识符和关键字。标识符是用来标识源程序中某个对象的名字的,这些对象可以是语句、数据类型、函数、变量、数组等等。C 语言是大小字敏感的一种高级语言,如果我们要定义一个定时器 1,可以写做"Timer1",如果程序中有"TIMER1",那么这两个是完全不同定义的标识符。标识符由字符串,数字和下划线等组成,注意的是第一个字符必须是字母或下划线,如"1Timer"是错误的,编译时便会有错误提示。有些编译系统专用的标识符是以下划线开头,所以一般不要以下划线开头命名标识符。标识符在命名时应当简单,含义清晰,这样有助于阅读理解程序。在 C51 编译器中,只支持标识符的前
32 位为有效标识,一般情况下也足够用了,除非你要写天书:P。
关键字则是编程语言保留的特殊标识符,它们具有固定名称和含义,在程序编写中不允许标识符与关键字相同。在 KEIL uVision2 中的关键字除了有 ANSI C 标准的 32 个关键字外还根据 51 单片机的特点扩展了相关的关键字。其实在 KEIL uVision2 的文本编辑器中编写
C 程序,系统可以把保留字以不同颜色显示,缺省颜色为天蓝色。(标准和扩展关键字请看附录一中的附表 1-1 和附表 1-2)
先看表 4-1,表中列出了 KEIL uVision2 C51 编译器所支持的数据类型。在标准 C 语
言中基本的数据类型为 char,int,short,long,float 和 double,而在 C51 编译器中 int 和 short 相同,float 和 double 相同,这里就不列出说明了。下面来看看它们的具体定义:
1. char 字符类型
char 类型的长度是一个字节,通常用于定义处理字符数据的变量或常量。分无符号字符类型 unsigned char 和有符号字符类型 signed char,默认值为 signed char 类型。
unsigned char 类型用字节中所有的位来表示数值,所可以表达的数值范围是 0~255。 signed char 类型用字节中最高位字节表示数据的符号,"0"表示正数,"1"表示负数,负数用补码表示。所能表示的数值范围是-128~+127。unsigned char 常用于处理 ASCII 字符或用于处理小于或等于 255 的整型数。
*正数的补码与原码相同,负二进制数的补码等于它的绝对值按位取反后加 1。
2. int 整型
int 整型长度为两个字节,用于存放一个双字节数据。分有符号 int 整型数 signed int 和无符号整型数 unsigned int,默认值为 signed int 类型。signed int 表示的数值范围是-32768~+32767,字节中最高位表示数据的符号,"0"表示正数,"1"表示负数。 unsigned int 表示的数值范围是 0~65535。
好了,先停一下吧,我们来写个小程序看看 unsigned char 和 unsigned int 用于延时的不同效果,说明它们的长度是不同的,呵,尽管它并没有实际的应用意义,这里我们学习它们的用法就行。依旧用我们上一课的最小化系统做实验,不过要加多一个电阻和 LED,如图 4-1。实验中用 D1 的点亮表明正在用 unsigned int 数值延时,用 D2 点亮表明正在用 unsigned char 数值延时。
图 4-1第 4 课实验用电路
我们把这个项目称为 TwoLED,实验程序如下:
#include <AT89X51.h> //预处理命令
void main(void) //主函数名
{
unsigned int a; //定义变量 a 为 unsigned int 类型 unsigned char b; //定义变量 b 为 unsigned char 类型
do
{ //do while 组成循环
for (a=0; a<65535; a++)
P1_0 = 0; //65535 次设 P1.0 口为低电平,点亮 LED
P1_0 = 1; //设 P1.0 口为高电平,熄灭 LED
for (a=0; a<30000; a++); //空循环
for (b=0; b<255; b++)
P1_1 = 0; //255 次设 P1.1 口为低电平,点亮 LED
P1_1 = 1; //设 P1.1 口为高电平,熄灭 LED
for (a=0; a<30000; a++); //空循环
}
while(1);
}
同样编译烧写,上电运行您就可以看到结果了。很明显 D1 点亮的时间长于 D2 点亮的时间。程序中的循环延时时间并不是很好确定,并不太适合要求精确延时的场合,关于这方面我们以后也会做讨论。这里必须要讲的是,当定义一个变量为特定的数据类型时,在程序使用该变量不应使它的值超过数据类型的值域。如本例中的变量 b 不能赋超出 0~255 的值,如 for (b=0; b<255; b++)改为 for (b=0; b<256; b++),编译是可以通过的,但运行时就会有问题出现,就是说 b 的值永远都是小于 256 的,所以无法跳出循环执行下一句 P1_1 = 1,从而造成死循环。同理 a 的值不应超出 0~65535。大家可以烧片看看实验的运行结果,同样软件仿真也是可以看到结果的。
3. long 长整型
long 长整型长度为四个字节,用于存放一个四字节数据。分有符号 long 长整型 signed long 和无符号长整型 unsigned long,默认值为 signed long 类型。signed int 表示的数值范围是-2147483648~+2147483647,字节中最高位表示数据的符号,"0"表示正数,"1"表示负数。unsigned long 表示的数值范围是 0~4294967295。
4. float 浮点型
float 浮点型在十进制中具有 7 位有效数字,是符合 IEEE-754 标准的单精度浮点型数据,占用四个字节。因浮点数的结构较复杂在以后的章节中再做详细的讨论。
5.*指针型
指针型本身就是一个变量,在这个变量中存放的指向另一个数据的地址。这个指针变量 -
要占据一定的内存单元,对不同的处理器长度也不尽相同,在 C51 中它的长度一般为 1~
3 个字节。指针变量也具有类型,在以后的课程中有专门一课做探讨,这里就不多说了。
6. bit 位标量
bit 位标量是 C51 编译器的一种扩充数据类型,利用它可定义一个位标量,但不能定义位指针,也不能定义位数组。它的值是一个二进制位,不是 0 就是 1,类似一些高级语言中的 Boolean 类型中的 True 和 False。
7. sfr 特殊功能寄存器
sfr 也是一种扩充数据类型,点用一个内存单元,值域为 0~255。利用它可以访问 51 单片机内部的所有特殊功能寄存器。如用 sfr P1 = 0x90 这一句定 P1 为 P1 端口在片内的寄存器,在后面的语句中我们用以用 P1 = 255(对 P1 端口的所有引脚置高电平)之类的语句来操作特殊功能寄存器。
*AT89C51 的特殊功能寄存器表请看附录二
8.sfr16 16 位特殊功能寄存器
sfr16 占用两个内存单元,值域为 0~65535。sfr16 和 sfr 一样用于操作特殊功能寄存器,所不同的是它用于操作占两个字节的寄存器,好定时器 T0 和 T1。
9. sbit 可录址位
sbit 同位是 C51 中的一种扩充数据类型,利用它可以访问芯片内部的 RAM 中的可寻址位或特殊功能寄存器中的可寻址位。如先前我们定义了
sfr P1 = 0x90; //因 P1 端口的寄存器是可位寻址的,所以我们可以定义
sbit P1_1 = P1^1;//P1_1 为 P1 中的 P1.1 引脚
//同样我们可以用 P1.1 的地址去写,如 sbit P1_1 = 0x91;
这样我们在以后的程序语句中就可以用 P1_1 来对 P1.1 引脚进行读写操作了。通常这些可以直接使用系统提供的预处理文件,里面已定义好各特殊功能寄存器的简单名字,直接引用可以省去一点时间,我自己是一直用的。当然您也可以自己写自己的定义文件,用您认为好记的名字。
关于数据类型转换等相关操作在后面的课程或程序实例中将有所提及。大家可以用所讲到的数据类型改写一下这课的实例程序,加深对各类型的认识。
第五课 常量
上一节我们学习了 KEIL C51 编译器所支持的数据类型。而这些数据类型又是怎么用在常量和变量的定义中的呢?又有什么要注意的吗?下面就来看看吧。晕!你还区分不清楚什么是常量,什么是变量。常量是在程序运行过程中不能改变值的量,而变量是可以在程序运行过程中不断变化的量。变量的定义可以使用所有 C51 编译器支持的数据类型,而常量的数据类型只有整型、浮点型、字符型、字符串型和位标量。这一节我们学习常量定义和用法,而下一节则学习变量。
常量的数据类型说明是这样的
1. 整型常量可以表示为十进制如 123,0,-89 等。十六进制则以 0x 开头如 0x34,-0x3B
等。长整型就在数字后面加字母 L,如 104L,034L,0xF340 等。
2. 浮点型常量可分为十进制和指数表示形式。十进制由数字和小数点组成,如
0.888,3345.345,0.0 等,整数或小数部分为 0,可以省略但必须有小数点。指数表示形式为[±]数字[.数字]e[±]数字,[]中的内容为可选项,其中内容根据具体情况可有可无,但其余部分必须有,如 125e3,7e9,-3.0e-3。
3. 字符型常量是单引号内的字符,如'a','d'等,不可以显示的控制字符,可以
在该字符前面加一个反斜杠"\"组成专用转义字符。常用转义字符表请看表 5-1。4. 字符串型常量由双引号内的字符组成,如"test","OK"等。当引号内的没有字符时,为空字符串。在使用特殊字符时同样要使用转义字符如双引号。在 C 中字符串常量是做为字符类型数组来处理的,在存储字符串时系统会在字符串尾部加上\o转义字符以作为该字符串的结束符。字符串常量"A"和字符常量'A'是不同的,
前者在存储时多占用一个字节的字间。
5. 位标量,它的值是一个二进制。
常量可用在不必改变值的场合,如固定的数据表,字库等。常量的定义方式有几种,下面来加以说明。
#difine False 0x0; //用预定义语句可以定义常量 #difine True 0x1; //这里定义 False 为 0,True 为 1
//在程序中用到 False 编译时自动用 0 替换,同理 True 替换为 1 unsigned int code a=100; //这一句用 code 把 a 定义在程序存储器中并赋值 const unsigned int c=100; //用 const 定义 c 为无符号 int 常量并赋值
以上两句它们的值都保存在程序存储器中,而程序存储器在运行中是不允许被修改的,所以如果在这两句后面用了类似 a=110,a++这样的赋值语句,编译时将会出错。
说了一通还不如写个程序来实验一下吧。写什么程序呢?跑马灯!对,就写这个简单易懂的吧,这个也好说明典型的常量用法。先来看看电路图吧。它是在我们上一课的实验电路的基础上增加 6 个 LED 组成的,也就是用 P1 口的全部引脚分别驱动一个 LED,电路如图 5 -1 所示。
新建一个 RunLED 的项目,主程序如下:
#include <AT89X51.H> //预处理文件里面定义了特殊寄存器的名称如 P1 口定义为 P1
void main(void)
{
//定义花样数据
cons tunsigned char design[32]={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; //在 C51 编程中因内存有限尽可能注意变量类型的使用
//尽可能使用少字节的类型,在大型的程序中很
受用
do{
for (b=0; b<32; b++)
{
for(a=0; a<30000; a++); //延时一段时间
P1 = design[b]; //读已定义的花样数据并写花样数据到 P1 口
}
}while(1);
}
程序中的花样数据可以自以去定义,因这里我们的 LED 要 AT89C51 的 P1 引脚为低电平才会点亮,所以我们要向 P1 口的各引脚写数据 O 对应连接的 LED 才会被点亮,P1 口的八个引脚刚好对应 P1 口特殊寄存器的八个二进位,如向 P1 口定数据 0xFE,转成二进制就是11111110,最低位 D0 为 0 这里 P1.0 引脚输出低电平,LED1 被点亮。如此类推,大家不难算出自己想要做的效果了。大家编译烧写看看,效果就出来,显示的速度您可以根据需要调整延时 a 的值,不要超过变量类型的值域就很行了。哦,您还没有实验板?那如何可以知道程序运行的结果呢?呵,不用急,这就来说说用 KEIL uVision2 的软件仿真来调试 IO 口输出输入程序。
图 5-1八路跑马灯电路
编译运行上面的程序,然后按外部设备菜单 Peripherals-I/O Ports-Port1 就打开 Port1 的调试窗口了,如图 5-3 中的 2。这时程序运行了,但我们并不能在 Port1 调试窗口上看到有会什么效果,这时我们可以用鼠标左击图 5-3 中 1 旁边绿色的方条,点一下就有一个小红方格在点一下又没有了,哪一句语句前有小方格程序运行到那一句时就停止了,就是设置调试断点,同样图 5-2 中的 1 也是同样功能,分别是增加/移除断点、移除所有断点、允许/禁止断点、禁止所有断点,菜单也有一样的功能,另外菜单中还有 Breakpoints 可打开断点设置窗口它的功能更强大,不过我们这里先不用它。我们在"P1 = design[b];"这一句设置一个断点这时程序运行到这里就停住了,再留意一下 Port1 调试窗口,再按图 5-2 中的 2 的运行键,程序又运行到设置断点的地方停住了,这时 Port1 调试窗口的状态又不同了。也就是说 Port1 调试窗口模拟了 P1 口的电平状态,打勾为高电平,不打勾则为低电平,窗口中 P1 为 P1 寄存器的状态,Pins 为引脚的状态,注意的是如果是读引脚值必须把引脚对应的寄存器置 1 才能正确读取。图 5-2 中 2 旁边的{}样的按钮分别为单步入,步越,步出和执行到当前行。图中 3 为显示下一句将要执行的语句。图 5-3 中的 3 是 Watches 窗口可查看各变量的当前值,数组和字串是显示其头一个地址,如本例中的 design 数组是保存在 code 存储区的首地址为 D:0x08,可以在图中 4 Memory 存储器查看窗口中的 Address 地址中打入 D:0x08 就可以查看到 design 各数据和存放地址了。如果你的 uVision2 没有显示这些窗口,可以在 View 菜单中打开在图 5-2 中 3 后面一栏的查看窗口快捷栏中打开。
图 5-2调试用快捷菜单栏
图 5-3各调试窗口
单片机中C语言的灵活运用,函数在单片机中的运用!
函数定义
函数是一个自我包含的完成一定相关功能的执行代码段。通常C语言的编译器会自带标准的函数库,这些都是一些常用的函数。标准函数已由编译器软件商编写定义,使用者直接调用就能了,而无需定义。但是标准的函数不足以满足使用者的特殊要求,因此C语言允许使用者根据需要编写特定功能的函数,要调用它必须要先对其进行定义。
定义的模式如下:
函数类型 函数名称(形式参数表)
函数类型 函数名称 (形式参数表)
{
函数体
}
函数类型是说明所定义函数返回值的类型。返回值其实就是一个变量,只要按变量类型来定义函数类型就行了。如函数不需要返回值函数类型能写作“void”表示该函数没有返回值。注意的是函数体返回值的类型一定要和函数类型一致,不然会造成错误。
函数名称的定义在遵循C语言变量命名规则的同时,不能在同一程序中定义同名的函数,这将会造成编译错误(同一程序中是允许有同名变量的,因为变量有全局和局部变量之分)。
形式参数是指调用函数时要传入到函数体内参与运算的变量,它可以是一个、几个或没有。当函数不需要形式参数时(即无参函数),括号内为空或写入“void”表示,但括号不能少。
函数体中能包含有局部变量的定义和程序语句,如函数要返回运算值则要使用return语句进行返回。若在函数的{}号中也能什么也不写,这就成了空函数。在一个程序项目中可以写一些空函数,在以后的修改和升级中能方便的在这些空函数中进行功能扩充。
函数的调用
(一)函数调用的一般说明
函数定义好以后,要被其它函数调用了才能被执行。C语言的函数是能相互调用的,但在调用函数前,必须对函数的类型进行说明,就算是标准库函数也不例外。
标准库函数的说明会被按功能分别写在不一样的头文件中,使用时只要在文件最前面用#include预处理语句引入相应的头文件。如前面使用的printf函数的说明是放在文件名为stdio.h的头文件中。
调用就是指一个函数体中引用另一个已定义的函数来实现所需要的功能,这个时候函数体称为主调用函数,函数体中所引用的函数称为被调用函数。主函数只是相对于被调用函数而言。
一个函数体中能调用数个其它的函数,这些被调用的函数同样也能调用其它函数,也能嵌套调用。但是在c51语言中有一个函数是不能被其它函数所调用的,它就是main主函数。
标准库函数只要用#include引入已写好说明的头文件,在程序就能直接调用函数了。如调用的是自定义的函数则要用如下形式编写函数类型说明:
类型标识符 函数的名称(形式参数表);
这样的说明方式是用在被调函数定义和主调函数是在同一文件中。也能把这些写到文件名.h的文件中用#include“文件名.h”引入。
如果被调函数的定义和主调函数不是在同一文件中的,则要用如下的方式进行说明,说明被调函数的定义在同一项目的不一样文件之上,这样说明的函数也能称为外部函数,定义如下:
extern类型标识符 函数的名称(形式参数表);
函数的定义和说明是完全不一样的,在编译的角度上看函数的定义是把函数编译存放在ROM的某一段地址上,而函数说明是告诉编译器要在程序中使用那些函数并确定函数的地址。如果在同一文件中被调函数的定义在主调函数之前,这个时候能不用说明函数类型。也就是说在main函数之前定义的函数,在程序中就能不用写函数类型说明了。能在一个函数体调用另一个函数(嵌套调用),但不允许在一个函数定义中定义另一个函数。还要注意的是函数定义和说明中的“类型、形参表、名称”等都要相一致。
(二)函数调用的一般形式
调用函数的一般形式如下:
函数名 (实际参数表)
“函数名”就是指被调用的函数。
实际参数表能为零或多个参数,多个参数时要用逗号隔开,每个参数的类型、位置应与函数定义时所的形式参数一一对应,它的作用就是把参数传到被调用函数中的形式参数,如果类型不对应就会产生一些错误。调用的函数是无参函数时不写参数,但不能省后面的括号。
下面我们看一下在实际应用中函数不同的调用方式:
1. 函数语句
例如printf(“Hello World! ”);
它以“Hello World! ”为参数调用printf这个库函数,在这里函数调用被看作了一条语句。
2. 函数参数
“函数参数”这种方式是指被调用函数的返回值当作另一个被调用函数的实际参数,如temp=StrToInt(CharB(16));CharB的返回值作为StrToInt函数的实际参数传递。
3. 函数表达式
例如temp=Count();
这个函数的调用作为一个运算对象出现在表达式中,称为函数表达式。例子中Count()返回一个int类型的返回值直接赋值给temp。注意的是这种调用方式要求被调用的函数能返回一个同类型的值,不然会出现不可预料的错误。
C51常用头文件
下面介绍一些常用的C51头文件:
absacc.h——包含允许直接访问8051不同存储区的宏定义;
assert.h——文件定义assert 宏,可以用来建立程序的测试条件;
ctype——字符转换和分类程序;
intrins.h——文件包含指示编译器产生嵌入式固有代码的程序的原型;
math.h——数学程序;
reg51.h——51的特殊寄存器;
reg52.h——52的特殊寄存器;
setjmp.h——定义jmp_buf类型和setjmp和longjmp程序的原型;
stdarg.h——可变长度参数列表程序;
stdlib.h——存储区分配程序;
stdio.h——标准输入和输出程序;
string.h——字符串操作程序、缓冲区操作程序。
对于常用的MCS-51单片机,必须包含reg51.h的头文件,因为该文件对51单片机的相关寄存器及位进行了定义,这样在程序中才可以使用这些资源。
reg51.h文件的具体内容如下:
#ifndef __REG51_H__
#define __REG51_H__
/* BYTE Register */ //单元定义
sfr P0 = 0x80;
sfr P1 = 0x90;
sfr P2 = 0xA 0 ;
sfr P3 = 0xB0;
sfr PSW = 0xD0;
sfr ACC = 0xE0;
sfr B = 0xF0;
sfr SP = 0x81;
sfr DPL = 0x82;
sfr DPH = 0x83;
sfr PCON = 0x87;
sfr TCON = 0x88;
sfr TMOD = 0x89;
sfr TL0 = 0x8A;
sfr TL1 = 0x8B;
sfr TH0 = 0x8C;
sfr TH1 = 0x8D;
sfr IE = 0xA8;
sfr IP = 0xB8;
sfr SCON = 0x98;
sfr SBUF = 0x99;
/* BIT Register */
/* PSW */ //位定义
sbit CY = 0xD7;
sbit AC = 0xD6;
sbit F0 = 0xD5;
sbit RS1 = 0xD4;
sbit RS0 = 0xD3;
sbit OV = 0xD2;
sbit P = 0xD0;
/* TCON */
sbit TF1 = 0x8F;
sbit TR1 = 0x8E;
sbit TF0 = 0x8D;
sbit TR0 = 0x8C;
sbit IE1 = 0x8B;
sbit IT1 = 0x8A;
sbit IE0 = 0x89;
sbit IT0 = 0x88;
/* IE */
sbit EA = 0xAF;
sbit ES = 0xAC;
sbit ET1 = 0xAB;
sbit EX1 = 0xAA;
sbit ET0 = 0xA9;
sbit EX0 = 0xA8;
/* IP */
sbit PS = 0xBC;
sbit PT1 = 0xBB;
sbit PX1 = 0xBA;
sbit PT0 = 0xB9;
sbit PX0 = 0xB8;
/* P3 */
sbit RD = 0xB7;
sbit WR = 0xB6;
sbit T1 = 0xB5;
sbit T0 = 0xB4;
sbit INT1 = 0xB3;
sbit INT0 = 0xB2;
sbit TXD = 0xB1;
sbit RXD = 0xB0;
/* SCON */
sbit SM0 = 0x9F;
sbit SM1 = 0x9E;
sbit SM2 = 0x9D;
sbit REN = 0x9C;
sbit TB8 = 0x9B;
sbit RB8 = 0x9A;
sbit TI = 0x99;
sbit RI = 0x98;
#endif
而absacc.h为对8051单片机的不同存储区的宏定义,具体如下:
#define CBYTE ((unsigned char volatile code *) 0) //定义程序存储器;
#define DBYTE ((unsigned char volatile data *) 0) //定义片内数据存储区;
#define PBYTE ((unsigned char volatile pdata *) 0) //定义页寻址空间;
#define XBYTE ((unsigned char volatile xdata *) 0) //定义片外数据存储区。
而intrins.h文件对指示编译器产生嵌入式代码,如空操作执行、位指令、栈操作指令等,该文件的具体内容如下:
extern void _nop_ (void); //空操作8051 NOP指令
extern bit _testbit_ (bit); //测试并清零位8051 JBC指令
extern unsigned char _cror_ (unsigned char,unsigned char); //字符循环左移
extern unsigned int _iror_ (unsigned int,unsigned char); //字符循环右移
extern unsigned long _lror_ (unsigned long,unsigned char); //整数循环右移
extern unsigned char _crol_ (unsigned char,unsigned char); //整数循环右移
extern unsigned int _irol_ (unsigned int,unsigned char); //整数循环左移
extern unsigned long _lrol_ (unsigned long,unsigned char); //长整数循环左移
extern unsigned char _chkfloat_(float); //测试并返回源点数状态
extern void _push_ (unsigned char _sfr); //压入堆栈
extern void _pop_ (unsigned char _sfr); //弹出堆栈
相关问答
51 单片机 一个工程中包含多个 C 文件怎么使用?要生成各个C文件的头文件,然后再主函数里加入这个.h的文件就好了要生成各个C文件的头文件,然后再主函数里加入这个.h的文件就好了
单片机C 编程中的return?一个函数可以定义为无返回值的,例如voidds1820_reset(void);在函数之前加入void,那么就不会有return,但是有的时候函数是需要计算出一个结果的,在调用这个函...
单片机 程序的“. c ”和“.h”格式都有什么区别?.c文件是整个程序中的一个或几个函数组成,在别的.C文件里可以调用它,不只是在主函数中。这样做可以增强程序的模块化,提高程序的可读性。当编制好一个模块时你...
C 语言驱动 单片机 实现位移控制是否可行?C语言驱动单片机实现位移控制是否可行?答案是C语言驱动单片机实现位移控制是可行的。“位移”也就是对单片机进行移位操作。知识点:移位操作(1)左移(2...C...
单片机 是如何 调用 函数的?看你要调用的函数是什么类型的咯。1、如果是这种缺省的,void可写可不写,如:fuction()//或者voidfuction()或者fuction(void)或者voidfuction(void...
单片机 怎么 调用 子函数?1.单片机可以通过调用子函数来实现特定功能。2.调用子函数的原因是为了将复杂的程序分解成多个小模块,提高代码的可读性和可维护性。子函数可以独立编写和测...
单片机C 语言,为什么定时器初始化程序一定要放在主程序里,不能被二次 调用 ?timer初始化与系统初始化你需要知道他们之间的关系,如果timer初始化对系统初始化存在依赖关系,那么timer初始化必须在系统初始化之后,或者添加到initSys()函数...
怎样51 单片机调用 两个子程序 - nCqfmaGy 的回答 - 懂得MOVR0,#08HMOVA,#0FEHLOOP:MOVP1,ALCALLDELRRADJNZR0,LOOP使用DJNZ指令。为0执行下一指令,不为0跳转。就可以调用两个子程序了估计是进入到倒计时....
单片机 中计算器程序 c 语言加减乘除肿么写?-ZOL问答#include
设计编写单片机程序是一个渐进的过程不可一蹴而就,毕竟单片机程序是与硬件有密切关系的,我们一般称为单片机程序叫底层硬件驱动程序。编写程序不但要对软件要熟...