单片机控制LED
LED简介
LED全称light emitting diode
LED本质是二极管,所以只有单向导通
LED在单片机上的连接
VCC是电源正极,电阻起限流作用电阻阻值为1000,单片机上显示为102,意思是是10*10^2
单片机用的是TTL电平,高电平为5v,低电平为0v
初始状态LED两端都会接高电平,所以只需要通过编辑程序,使得左边电平变为低电平即能点亮LED
其各个LED连接于单片机的位置为:
单片机的寄存机为8位,每一位可以控制其电平的正负
LED连接的寄存器名为P2
CPU可以访问寄存器去写值,然后使寄存器发送信息给驱动器,然后就可以控制完成我们想要的功能
若在寄存器的一位上写1则为高电平,写0则为低电平
点亮LED
只需要往P2上写入数字即可控制LED闪烁
#include <REGX52.H>
void main()
{
P2=0xFE;
while(1){
}
}
16进制的FE表示二进制的1111 1110,意思是第一个LED为0,即低电平,可以点亮,其他LED为1,即高电平,无法点亮
如果想要点亮其他LED,或者更多的LED,就只需要改变P2的值即可
除此之外,在#include <REGX52.H>头文件中,还可以仅输入单个LED的值,而不需要输入十六进制
#include <REGX52.H>
void main()
{
while(1)
{
P2_0=0;//可以让第一个LED点亮
P2_1=0//可以让第二个LED点亮,其他以此类推,如果写等于1则为高电平无法点亮
}
}
如何延时程序
在stc上的软件延时计算器中可以生产延时函数,系统频率的选择由单片机金振的频率来决定
但这个代码会有一个函数 __ nop__(),这个函数需要一个头文件:#include<INTRINS.H>
__ nop__()是一个空语句
然后使用这个函数,再在主函数里调用
制造可变延时函数
在stc中选择一个延时1毫秒的函数然后更改成
void Delay(unsigned int xms) //@12.000MHz
{
unsigned char i, j;
while(xms){
i = 2;
j = 239;
do
{
while (--j);
} while (--i);
xms--;
}
}
在函数定义时在delay函数中添加参数xms,然后把复制好的代码放在while循环里,使得1毫秒循环xms次,即可使这个函数的延时可以自己输入
使LED闪烁
#include <REGX52.H>
#include <INTRINS.H>
void Delay500ms() //@12.000MHz
{
unsigned char i, j, k;
_nop_();
i = 4;
j = 205;
k = 187;
do
{
do
{
while (--k);
} while (--j);
} while (--i);
}
void main()
{
while(1){
P2=0xFE;//1111 1110
Delay500ms();
P2=0xFF;//1111 1111
Delay500ms();
}
}
这样即可让延时LED闪烁间隔,使得每次执行到Delay500ms();之后都要等0.5s才可以执行下一个语句,从而使得第一个LED能够一秒闪烁一次
LED流水灯
#include <REGX52.H>
#include <intrins.h>
void Delay(unsigned int xms) //@12.000MHz
{
unsigned char i, j;
while(xms){
i = 2;
j = 239;
do
{
while (--j);
} while (--i);
xms--;
}
}
void main()
{
while(1)
{
P2=0xFE;//1111 1110
Delay(500);
P2=0xFD;//1111 1101
Delay(500);
P2=0xFB;//1111 1011
Delay(500);
P2=0xF7;//1111 0111
Delay(500);
P2=0xEF;//1110 1111
Delay(500);
P2=0xDF;//1101 1111
Delay(500);
P2=0xBF;//1011 1111
Delay(500);
P2=0x7F;//0111 1111
Delay(500);
}
}
其原理与让LED闪烁类似都是让代码执行时延迟
独立按键控制LED
独立按键
有4个独立按键,每一个独立按键有一端都连在负极上,另一端各自连在不同的口
由于设计时k1与k2接反,所以需要主要的是第一个按键为P3_1,第二个按键为P3_0,第三个按键为P3_2,第三个按键为P3_3
按键在不按下时默认值为1,若按下第一个键,则P3_10,若按下第二个键,则P3_00,以此类推
独立按键按下与松开时存在按键抖动
那么就需要消抖,只需要判断按下之后,延时20ms,松开后又延时20ms,然后再进行其他操作,即可消抖
if(P3_1==0){
Delay(20);
while(P3_1==0);
Delay(20);
控制亮灭
由上知只需要判断按键是否按下,在按下时执行代码为LED点亮,松开执行代码LED熄灭即可
#include <REGX52.H>
void main()
{
while(1)
{
if(P3_0==0){
P2_0=0;
}else{
P2_0=1;
}
}
}
控制状态
这里需要做到按下去一次就点亮,按下去第二次就熄灭,所以如果不消抖,则可能会导致按一次就检测出按了不止一次,则无法准确表达。并且每次按下之后,LED的状态都应该为之前状态的相反状态,所以需要使用按位取反,因此代码为:
#include <REGX52.H>
void Delay(unsigned int xms)
{
unsigned char i, j;
while(xms){
i = 2;
j = 239;
do
{
while (--j);
} while (--i);
xms--;
}
}
void main()
{
while(1){
if(P3_1==0){
Delay(20);
while(P3_1==0);
Delay(20);
P2_0=~P2_0;
}
}
}
显示二进制
由于P2的初始状态为1111 1111,且其点亮的条件是0所以想要灯像二进制一样亮,只需要每一次按下都让P2减一,所以代码为:
#include <REGX52.H>
void Delay(unsigned int xms) //@12.000MHz
{
unsigned char i, j;
while(xms){
i = 2;
j = 239;
do
{
while (--j);
} while (--i);
xms--;
}
}
void main()
{
while(1){
if(P3_1==0){
Delay(20);
while(P3_1==0);
Delay(20);
P2--;
}
}
}
控制LED移位
想要按第一个键往左移,则需要使得如:P2从1110 1111到1101 1111
想要按第一个键往右移,则需要使得如:P2从1110 1111到1111 0111
且要使0111 1111与1111 1110能够按一次左或右就能转换
则无法直接使用P2=P2<<1来实现,而需要一个中间变量LEDNum
#include <REGX52.H>
void Delay(unsigned int xms) //@12.000MHz
{
unsigned char i, j;
while(xms){
i = 2;
j = 239;
do
{
while (--j);
} while (--i);
xms--;
}
}
unsigned char LEDNum=0;
void main()
{
P2=~0x01;
while(1)
{
if(P3_0==0)
{
Delay(20);
while(P3_0==0);
Delay(20);
LEDNum++;
if(LEDNum==8)
{
LEDNum=0;
}
P2=~(0x01<<LEDNum);
}
if(P3_1==0)
{
Delay(20);
while(P3_1==0);
Delay(20);
if(LEDNum==0)
{
LEDNum=7;
}else
{
LEDNum--;
}
P2=~(0x01<<LEDNum);
}
}
}