【51单片机】点亮LED之经典流水灯
开发环境
- 开发板:普中51-单核-A2
- 单片机:STC89C52RC(双列直插40引脚 DIP40)
- Keil uVision5 v9.61
最新版破解方法自行百度,相关文档和视频资料很多,我自己将这一操作记录下来当做博客发布,CSDN以版权问题审核不通过。
入门学习单片机的第一个教学程序基本都是点灯,无论是51单片机还是STM32,点灯程序在单片机学习的地位无异于其他编程语言学习的"hello,world"程序。点灯程序简单且效果明显,很容易上手,学过单片机的人人都能成为点灯大师。
思路分析
- 查看开发板原理图,找到LED模块
如图,D1到D8这8个发光二极管的阳极都连接到了VCC(开发板电源正极),这叫做共阳连接,阴极又分别串联一个电阻连接到P2口的0到7这8个引脚。根据二极管导通原理,要让某个LED亮,对应的引脚就业输出低电平(即输出0)。
编码
点亮一个LED
// Version 1: 点亮D1这一个LED
#include <REGX52.H>
void main()
{
P2=0XFE;
while(1)
{
}
}
8个LED呈现亮灭亮灭…的状态
// Version 2: 8个LED交替点亮
#include <REGX52.H>
int exitFlag = 0;
int main()
{
while(1)
{
P2 = 0x55;
if (exitFlag == 1)
{
goto EXIT;
}
}
EXIT:
return 0;
}
大部分51单片机教程代码的main函数都是声明为void类型,但是学过C语言的都知道标准的C语言的main函数都是int型的,return 0;是返回给操作系统,告诉系统程序正常执行完毕。但是现在写的点灯程序51单片机没有上操作系统,直接写void main也是有道理的,写成int main(void)纯属我个人习惯,而单片机return 0;又不能写作while(1)里面,否则执行一次就退出,死循环不就失效了,写在循环后面编译又会有一个警告,如下图:
因此只能使用goto语句加标号,去除这个警告,这也纯属个人习惯。虽然初学C语言的时候,被告知少用goto语句,因为会使得程序执行流程非常混乱,但实际上可以灵活应用,只要有把握就好,尽信书,不如无书,goto被发明出来就一定有用得到的地方。
流水灯
延时函数
实现流水灯,就不得不提延时函数了,因为单片机执行程序太快,不加延时看不出流水灯效果,需要加延时,让人眼观察到才行。而STC-ISP就可以生成软件延时代码。
从原理图可以看到开发板载的晶振是12Mhz,所以系统频率也选择12Mhz。指令集选STC-Y1,不同指令集件右边描述适用于哪些系列,正如右边所显示:STC-Y1指令集使用于本开发板的单片机系列。
将生成的代码拷贝到Keil的源代码中。
编写main函数实现主要功能
#include <REGX52.H>
void Delay100ms();
int exitFlag = 0;
int main()
{
while(1)
{
P2=0xFE;//1111 1110
Delay100ms();
P2=0xFD;//1111 1101
Delay100ms();
P2=0xFB;//1111 1011
Delay100ms();
P2=0xF7;//1111 0111
Delay100ms();
P2=0xEF;//1110 1111
Delay100ms();
P2=0xDF;//1101 1111
Delay100ms();
P2=0xBF;//1011 1111
Delay100ms();
P2=0x7F;//0111 1111
Delay100ms();
if (exitFlag == 1)
{
goto EXIT;
}
}
EXIT:
return 0;
}
/* 延时100ms的函数 */
void Delay100ms() //@12.000MHz
{
unsigned char i, j;
i = 195;
j = 138;
do
{
while (--j);
} while (--i);
}
程序升级-使用循环
#include <REGX52.H>
void Delay100ms();
int exitFlag = 0;
int main()
{
int i;
P2 = ~0x01; // 初始值,用于初始显示
while(1)
{
for(i = 0; i < 7; i++)
{
P2 = ~(0x01 << i);
Delay100ms();
}
if (exitFlag == 1)
{
goto EXIT;
}
}
EXIT:
return 0;
}
/* 延时100ms的函数 */
void Delay100ms() //@12.000MHz
{
unsigned char i, j;
i = 195;
j = 138;
do
{
while (--j);
} while (--i);
}
任意毫秒的延时函数的实现
以上代码是以固定100ms的周期闪烁,很无聊!!其实可以用STC-ISP生成1ms延时函数,并利用这个1ms延时函数生成延时任意毫秒的函数方法。
#include <REGX52.H>
void Delay1ms();
void DelayXms(int xms);
int exitFlag = 0;
int main()
{
int i;
P2 = ~0x01; // 初始值,用于初始显示
while(1)
{
for(i = 0; i < 7; i++)
{
P2 = ~(0x01 << i);
DelayXms(500); // 以500ms为周期运行流水灯
}
if (exitFlag == 1)
{
goto EXIT;
}
}
EXIT:
return 0;
}
/* 延时1ms */
void Delay1ms() //@12.000MHz
{
unsigned char i, j;
i = 2;
j = 239;
do
{
while (--j);
} while (--i);
}
// 传入多少参数,就延时多少毫秒
void DelayXms(int xms)
{
int i;
for(i = 0; i < xms; i++)
{
Delay1ms();
}
}
这样之前的延时100ms的函数就可以用
DelayXms(100);
代替了,之前的延时100ms函数的代码就可以删除掉了。因为Keil C51的编译器检测到某些函数定义了却没被调用会报警告,如果对程序编译完报的warning很难受(强迫症),可以删除实际上没有被调用的无用函数,毕竟嵌入式开发非常看重内存资源,这一点没用的小代码也会占用一定的空间。