51单片机系列之---点亮LED灯

实验介绍:利用51单片机,通过对CPU中相关寄存器的改写,实现对开发板上的LED灯实现控制。

实验设备:普中51-实验开发板

一、硬件知识

LED 是 Light Emitting Diode 的缩写,中文名称为发光二极管。发光二极管可高效地将电能转化为光能,可以将电信号转换为光信号的发光器件

特点:功耗低、高亮度、色彩鲜艳、抗震动、寿命长

直插式LED

长正短负(长的代表正极,短的代表负极)

贴片式

LED(发光二极管)两端存在电压差,有一定的电流流过时会亮起。电流可以理解为水流,电压差可以理解为水位差,当两个点水位高度不一样时,水流会从高水位流向低水位。 

这是一个LED在电路图中的符号 ,其中两个箭头使其与二极管区分,大屁股表示阳极(+),墙表示阴极(-)

二、原理图

下图为开发板上LED模块的电路图,如图所示,8个LED灯左端(正极)接入VCC,右端(负极)各接入一个电阻,此处的电阻起到的作用是限流,防止通路上的电流过大而烧毁LED,在电阻的右端,分别连接P20-P27接口。

一般插件LED电流是20ma左右,压降:红/黄色1.8V, 蓝/白色 3V, 实际电压要看 LED 规格书。

一般贴片LED电流是5ma左右,压降:红/绿/橙色1.8V,蓝/白色 3V

例如:供电电压是3.3V,黄色贴片LED

根据V = I * R ,则R= (3.3 - 1.8) /0.005 (5ma = 0.005), 所以 R = 300 欧姆.

很多时候你看到别人设计的电路中,LED串联的电阻去到几百欧或几千欧都有,是设计错了吗?

实际上这是非常合理的,因为大多数电路中,LED只是一个提示灯,对亮度没有要求,反而希望把功耗降低,所以需要增大限流电阻来实现超低电流,像产品中的贴片LED去到0.5ma也是能看清楚灯光的。

在原理图的单片机核心部分,我们可以看到上述的P20-P27端,与单片机的P2.0-P2.7连接。

通过分析电路硬件图以及原理图,我们得到相应的操作,要是LED(以D1为例)发亮,现在正极已经接上VCC,只要使得负极接地(0V)处于低电平就能够点亮,其负极与单片机的P2.0相连。那么现在的目标就是使P2.0端口输出低电平即可,将问题转化为控制单片机引脚输出高低电平的问题。

        在单片机中,CPU通过控制相关寄存器,寄存器通过驱动器来控制某个端子的高低电平,这也就是说,我们只要配置相应的CPU寄存器(P2寄存器)来实现我们的功能。

三、代码分析

1.点亮一颗LED灯

#include "reg52.h"//引入8052系列微控制器的头文件

sbit led1 = P2^0;//为P2寄存器的第0位创建了一个单独的位变量led1

void main(){
    while(1){//循环使芯片不断输出,否则瞬间运行完代码,我们无法观察
        led1 = 0;//0输出低电平,1输出高电平,这里0输出低电平形成电压差从而使LED灯亮起
    }
}

 2.闪烁LED

#include <reg52.h>

sbit led = P2^7;

//带参延时函数
void delay_ms(unsigned int xms)   //@12MHz
{
    unsigned int i, j;
    for(i=xms;i>0;i--)
    {
        for(j=124;j>0;j--)
        {}
    }
}

void main()
{
    while(1)
    {
        led = 0;
        delay_ms(500);
        led = 1;
        delay_ms(500);
    }
}

 通过添加一个不断递增的函数来控制led灯亮与灭的间隔

3.流水灯

(1)第一种

#include <reg52.h>

sbit led1 = P2^0;
sbit led2 = P2^1;
sbit led3 = P2^2;
sbit led4 = P2^3;
sbit led5 = P2^4;
sbit led6 = P2^5;
sbit led7 = P2^6;
sbit led8 = P2^7;

void delay_ms(unsigned int xms) 
{
    unsigned int i, j;
    for(i=xms;i>0;i--)
    {
        for(j=124;j>0;j--)
        {}
    }
}

void main(){
	while(1){
		led1=0;led2=1;led3=1;led4=1;led5=1;led6=1;led7=1;led8=1;
		delay_ms(500);
		led1=1;led2=0;led3=1;led4=1;led5=1;led6=1;led7=1;led8=1;
		delay_ms(500);
		led1=1;led2=1;led3=0;led4=1;led5=1;led6=1;led7=1;led8=1;
		delay_ms(500);
		led1=1;led2=1;led3=1;led4=0;led5=1;led6=1;led7=1;led8=1;
		delay_ms(500);
		led1=1;led2=1;led3=1;led4=1;led5=0;led6=1;led7=1;led8=1;
		delay_ms(500);
		led1=1;led2=1;led3=1;led4=1;led5=1;led6=0;led7=1;led8=1;
		delay_ms(500);
		led1=1;led2=1;led3=1;led4=1;led5=1;led6=1;led7=0;led8=1;
		delay_ms(500);
		led1=1;led2=1;led3=1;led4=1;led5=1;led6=1;led7=1;led8=0;
		delay_ms(500);
	}
}

(2)第二种

#include <reg52.h>

//延时函数
void delay_ms(unsigned int xms) 
{
    unsigned int i, j;
    for(i=xms;i>0;i--)
    {
        for(j=124;j>0;j--)
        {}
    }
}

void main() {
    unsigned char i;
    while (1) {
        for (i = 0; i < 8; i++) {
            P2 = (0xFF ^ (1 << i));  // 将 1 左移 i 位,然后取反,点亮第 i 个 LED
            delay_ms(500);        // 延时 500 毫秒
        }
    }
}

(3)第三种

#include "reg52.h"
#include "intrins.h"

typedef unsigned int u16;	//对系统默认数据类型进行重定义
typedef unsigned char u8;

#define LED_PORT	P2	//使用宏定义P2端口

void delay_10us(u16 ten_us)
{
	while(ten_us--);	
}

void main()
{	
   	u8 i=0;

	LED_PORT=~0x01;
	delay_10us(50000);
	while(1)
	{
		for(i=0;i<7;i++)	 //将led左移一位
		{									  
			LED_PORT=_crol_(LED_PORT,1);
			delay_10us(50000); 	
		}
		for(i=0;i<7;i++)	//将led右移一位
		{
			LED_PORT=_cror_(LED_PORT,1);
			delay_10us(50000);	
		}	
	}		
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值