51单片机红外通信控制LED灯(LCD1602显示)
家好,再见到你,离上次开始DS18B20传感器的文章已经过去了一个星期了,这期我将给大家带来,基于芯片和红外通信实验红外接收探头
红外通信
-
- 51单片机红外通信控制LED灯(LCD1602显示)
- 红外遥控电路的组成
-
- 调制和解调信号 及 和解码红外编码协议
- NEC协议
-
-
- 数据码
-
红外遥控电路的组成
在我们的生活中,红外遥控系统由发射装置和接收装置组成,即(包括键盘电路、红外编码芯片、电源(我们今天用的是一个小按钮电池)、红外发射电路)和(智能灯、风扇、空调、电视等)
红外接收设备可由红外接收电路、红外解码芯片、电源和应用电路组成。今天,我们将开发板和红外接收探头组合成红外接收电路用作解码芯片 此为红外通信流程图,其实很简单,并没有那么复杂,也就是遥控器发射红外线,被控制的物品接收红外线做我们人类编程程序中的任务。红外线肉眼不可见,但只要我们拿手机摄像头对准遥控器的发射端,就能看见一个紫光 这是我们今天要使用的遥控器
我们要用的长这个样子 我用的清翔51单片机将是HS0038的out口接到了P3^2这个IO我们熟悉的嘴,也就是说INT0(外部中断0),所以要通过编写中断程序来接收红外携带的1和0。
因此,我们不需要红外通信sbit IO只需充分利用中断服务程序即可。
调制和解调信号 及 和解码红外编码协议
通常,为了更好地传输信号,发送端将基带二进制信号调制为脉冲串信号,并通过红外发射管发射
这样,单片机就可以更容易地区分数据0和1的脉冲时间,以确定它是数据0还是数据1
所以,我们介绍一个NEC协议,让大家更好的了解如何接收和识别红外发射的信号
NEC协议
:以下是发射端的方波图,接收端正好相反,数据传输从最低水平开始,所以我们需要编写正确的程序来识别数据 你也可以从这张照片中看到,我们不需要引导码。我们主要需要的是数据码,通过识别数据码来控制单片机上的小灯或其他东西 其中,引导码高电平约9万us 低电平约4500us 左右; 用户码16 位,数据码16 位,共32位; 数据0 高电平约560us +低电平约560us”表示。 数据1 高电平约560us+低电平约1680us”表示。 因此,我们可以通过不同的脉冲宽度来识别是0还是1
在这里,我们需要使用定期中断函数和中断服务程序来增加时间,这样我们就可以接收正确的脉冲宽度并识别它们
void Init_timer0() {
EA = 1; TR0 = 1; TMOD = 0X02; //八位自动重装 ET0 = 1; TH0 = 0; TL0 = 0; } void timer0() interrupt 1 {
time_num ; //256us }
这是我的定期中断函数和中断服务程序,每256次us会进一次中断函数,让我的计时变量time_num加1,这样我们就可以很容易地接收到正确的脉冲宽度。
:脉冲宽度为1.12ms数据1的脉冲宽度为2.25ms 我们只需要2250/256 = 8.78 我们只需要判断我们是否收到超过7个数据,如果超过7个数据,我们认为我们收到的数据是1,这样我们就可以成功地收到32个数据,即四个字节。
接下来是看代码的时候了。学习红外通信的小伙伴不应该缺乏阅读代码和理解代码的能力。当然,如果你不明白,你可以通过私人信件发送博客作者。我会一一回答。
#include<reg52.h> #define uchar unsigned char #define uint unsigned int
sbit RS = P3^5;
sbit RW = P3^6;
sbit EN = P3^4;
sbit dula = P2^6;
sbit wela = P2^7;
sbit LED1 =P1^0;
uchar time_num,extern_num;
uchar timerecord[33];
uchar cord[4];
uchar flag_ok;
uchar count;
/* LCD1602 */
void Read_Busy()
{
uchar busy;
P0 = 0XFF; //将P0复位
RS = 0;
RW = 1;
do
{
EN = 1;
busy = P0;
EN = 0; //以便下一次产生上升沿
}while(busy & 0x80);
}
void LCD_Write_cmd(uchar cmd) //写入操控lcd的指令
{
Read_Busy();
RS = 0;
RW = 0;
P0 = cmd;
EN = 1;
EN = 0;
}
void LCD_Write_dat(uchar dat)
{
Read_Busy();
RS = 1;
RW = 0;
P0 = dat;
EN = 1;
EN = 0;
}
void LCD_Init()
{
LCD_Write_cmd(0x38);
LCD_Write_cmd(0x0c);
LCD_Write_cmd(0x06);
LCD_Write_cmd(0x01);
}
/* LCD1602 */
void Init_INT0()
{
EA = 1;
EX0 = 1;
IT0 = 1;
}
void Init_timer0()
{
EA = 1;
TR0 = 1;
TMOD = 0X02; //八位自动重装
ET0 = 1;
TH0 = 0;
TL0 = 0;
}
void processing_jiema()
{
uchar i,j,k = 1,jiema;
for(j=0;j<4;j++)
{
for(i=0;i<8;i++)
{
jiema >>= 1;
if(timerecord[k] > 6)
{
jiema|=0x80;
}
k++;
}
cord[j] = jiema;
// jiema = 0; //可写可不写
}
}
void LCD1602_Display()
{
uchar i;
LCD_Write_cmd(0x80+0x04); //第一行第五个
for(i=0;i<4;i++)
{
if(cord[i]/16<10)
{
LCD_Write_dat(cord[i]/16 + 0x30);
}
else
{
LCD_Write_dat(cord[i]/16 + 0x37);
}
if(cord[i]%16<10)
{
LCD_Write_dat(cord[i]%16 + 0x30);
}
else
{
LCD_Write_dat(cord[i]%16 + 0x37);
}
}
}
void main()
{
Init_INT0();
Init_timer0();
LCD_Init();
dula = 0;
wela = 0;
while(1)
{
if(flag_ok == 1)
{
processing_jiema();
LCD1602_Display();
flag_ok = 0;
}
switch(cord[2])
{
case 0x0c:LED1 = 0; break;
case 0x18:LED1 = 1; break;
}
}
}
void INT_0() interrupt 0
{
extern_num++;
if(extern_num == 1)
{
time_num = 0;
}
else
{
if(time_num > 32) //起始码判断
{
count = 0;
}
timerecord[count] = time_num; //第一个是起始码,不需要
time_num = 0;
count++;
if(count == 33)
{
extern_num = 0;
flag_ok = 1;
}
}
}
void timer0() interrupt 1
{
time_num++; //256us
}
/* 1码的脉冲宽度为2.25ms 0码的脉冲宽度为1.12ms 起始码的脉冲宽度为9ms */
在我的代码中,我用了LCD1602来显示接收到的32位数据,这样我就可以知道按键的各个部分的数据码是什么,通过数据码的识别来控制小灯的亮和熄灭。
数据码
按键名称 | 数据码 |
---|---|
0 | 0x16 |
1 | 0x0c |
2 | 0x18 |
3 | 0x5e |
4 | 0x08 |
5 | 0x1c |
6 | 0xaa |
7 | 0x42 |
8 | 0x52 |
9 | 0xab |
- | 0x07 |
+ | 0x15 |
eq | 0x09 |
next | 0x40 |
prev | 0x44 |
play | 0x87 |
感谢大家的收看!喜欢博主的就点个赞还有点个关注吧!