求知 文章 文库 Lib 视频 iPerson 课程 认证 咨询 工具 讲座 Modeler   Code  
会员   
要资料
 
追随技术信仰

随时听讲座
每天看新闻
 
 
单片机教程
第一章 单片机入门
1.单片机简介
2.开发板选择
3.前期准备
第二章 LED及入门
1.初识原理图
2.程序点灯
3.实际LED硬件连接
4.点亮LED
5.闪烁的LED
6.软件调试查看运行时间
7.延时1秒
8.函数封装
9.函数传参调用
10.流水灯
11.数组与移位
第三章 蜂鸣器
1.蜂鸣器原理
2.无源蜂鸣器鸣叫
3.无源蜂鸣器题目
4.无源蜂鸣器+LED
第四章 数码管
1.数码管原理
2.数组&数码管
3.数码管&LED
4.多个数码管显示
5.同时显示不同的数字
6.高位不显示0
7.数码管小数点
8.分钟秒表
第五章 独立按键
1.按键入门
2.按键&蜂鸣器&数码管
3.按键猜想
4.按键用法
5.按键模式
6.不支持连按
7.静态变量
8.支持连按与全局变量
9.再次优化不支持连按的代码
10.流水灯的优化解说
11. 代码对比
12.双模式函数封装
13.第一阶段综合例程(上)
14.第一阶段综合例程(下)
15.第一阶段的总结
第六章 多文件编程
1.多文件编写
2.模块初始化
3.数码管显示函数
4.多文件编程首次测试
5.带返回值的函数
6.新按键程序
7.最终按键程序
8.最终的function文件
第七章 外部中断
1.寄存器
2.中断函数
3.进一步理解中断函数
4.中断的实验现象
5.外部中断测脉冲个数
第八章 定时器
1.定时器概念
2.定时器工作模式
3.定时时长的做法
4.定时器简单运用
5.定时器工作模式2
6.定时器中断函数的使用
7.定时器初始化新写法
8.隐形漏洞
9.代码参考
10.微调定时精确时间
11.单独文件封装
12.输入捕获
13.数据类型强制转换
14.定时器&数码管扫描显示
15.呼吸灯
第九章 舵机与超声波模块
1.舵机入门
2.舵机与按键
3.超声波模块
第十章 串口通信
1.串口通信入门
2.简洁式串口通信
3.详细理解ASCII码
4.串口printf系列函数
第十一章 1602液晶屏
1.液晶屏代码讲解
2.指向数组的指针
3.液晶屏代码单独文件
4.各功能代码的运用
第十二章 IIC通信
1.IIC入门代码讲解
2.EEPROM简单使用
3.IIC&EEPROM合成文件
第十三章 红外遥控与温度传感器
1.红外遥控
2.代码文件和测试
3.温度传感器代码解析Ⅰ
4.温度传感器代码解析Ⅱ
5.温度传感器代码独立文件
第十四章 AD与DA
1.AD与DA合成文件
第十五章 混合例程
1.电位器控制舵机
2.遥控器控制舵机
3.温度传感器与串口
4.模拟倒车雷达报警
5.再次熟悉串口
6.串口&液晶屏Ⅰ
7.串口&液晶屏Ⅱ
8.串口&液晶屏Ⅲ
9.串口&EEPROM
10.实践操作
11.结束语
 
 
目录
代码文件和测试
来源:C语言网    作者:继承叔    编辑:Alice(火龙果软件)
498 次浏览
3次  

创建好“infrared.c”和“infrared.h”文件。

1.infrared.c的代码

#include <reg52.h>
#include <infrared.h>
   
unsigned char irflag = 0; //红外接收标志,收到一帧正确数据后置1
unsigned char ircode[4];  //红外代码接收缓冲区
   
/* 初始化红外接收功能 */
void InitInfrared()
{
    IR_INPUT = 1;  //确保红外接收引脚被释放
    TMOD &= 0x0F;  //清零T1的控制位
    TMOD |= 0x10;  //配置T1为模式1
    TR1 = 0;       //停止T1计数
    ET1 = 0;       //禁止T1中断
    IT1 = 1;       //设置INT1为负边沿触发
    EX1 = 1;       //使能INT1中断
}
  
/* 获取当前高电平的持续时间 */
unsigned int GetHighTime()
{
    TH1 = 0;                //清零T1计数初值
    TL1 = 0;
    TR1 = 1;                //启动T1计数
    while (IR_INPUT)        //红外输入引脚为1时循环检测等待,变为0时则结束本循环
    {
        if (TH1 >= 0x40)
        {                   //当T1计数值大于0x4000,即高电平持续时间超过约18ms时,
            break;          //强制退出循环,是为了避免信号异常时,程序假死在这里。
        }
    }
    TR1 = 0;                //停止T1计数
    
    return (TH1*256 + TL1); //T1计数值合成为16bit整型数,并返回该数
}
  
/* 获取当前低电平的持续时间 */
unsigned int GetLowTime()
{
    TH1 = 0;                //清零T1计数初值
    TL1 = 0;
    TR1 = 1;                //启动T1计数
    while (!IR_INPUT)       //红外输入引脚为0时循环检测等待,变为1时则结束本循环
    {
        if (TH1 >= 0x40)
        {                   //当T1计数值大于0x4000,即低电平持续时间超过约18ms时,
            break;          //强制退出循环,是为了避免信号异常时,程序假死在这里。
        }
    }
    TR1 = 0;                //停止T1计数
    
    return (TH1*256 + TL1); //T1计数值合成为16bit整型数,并返回该数
}
  
/* INT1中断服务函数,执行红外接收及解码 */
void EXINT1_ISR() interrupt 2
{
    unsigned char i,j;
    unsigned char byt;
    unsigned int time;
      
    //接收并判定引导码的9ms低电平
    time = GetLowTime();
    if ((time<7833) || (time>8755))  //时间判定范围为8.5~9.5ms,
    {                                //超过此范围则说明为误码,直接退出
        IE1 = 0;                     //退出前清零INT1中断标志
        return;
    }
    //接收并判定引导码的4.5ms高电平
    time = GetHighTime();
    if ((time<3686) || (time>4608))  //时间判定范围为4.0~5.0ms,
    {                                //超过此范围则说明为误码,直接退出
        IE1 = 0;
        return;
    }
  
    //接收并判定后续的4字节数据
    for (i=0; i<4; i++)     //循环接收4个字节
    {
        for (j=0; j<8; j++) //循环接收判定每字节的8个bit
        {
            //接收判定每bit的560us低电平
            time = GetLowTime();
            if ((time<313) || (time>718))        //时间判定范围为340~780us,
            {                                    //超过此范围则说明为误码,直接退出
                IE1 = 0;
                return;
            }
            //接收每bit高电平时间,判定该bit的值
            time = GetHighTime();
            if ((time>313) && (time<718))        //时间判定范围为340~780us,
            {                                    //在此范围内说明该bit值为0
                byt >>= 1;                       //因低位在先,所以数据右移,高位为0
            }
            else if ((time>1345) && (time<1751)) //时间判定范围为1460~1900us,
            {                                    //在此范围内说明该bit值为1
                byt >>= 1;   //因低位在先,所以数据右移,
                byt |= 0x80; //高位置1
            }
            else             //不在上述范围内则说明为误码,直接退出
            {
                IE1 = 0;
                return;
            }
        }
        ircode[i] = byt;     //接收完一个字节后保存到缓冲区
    }
    irflag = 1;              //接收完毕后设置标志
    IE1 = 0;                 //退出前清零INT1中断标志
}

2.infrared.h的代码

因为主函数中只需要使用“InitInfrared();”函数来初始化,而“unsigned int GetHighTime()”和“unsigned int GetLowTime()”都是只在“void EXINT1_ISR() interrupt 2”中使用,所以头文件不用声明这三部分的函数。

#ifndef __INFRARED_H__
#define __INFRARED_H__
   
sbit IR_INPUT = P3^3;           //红外接收引脚
extern unsigned char irflag;    //红外接收标志,收到一帧正确数据后置1
extern unsigned char ircode[4]; //红外代码接收缓冲区
   
void InitInfrared();            //初始化红外接收功能
   
#endif

 

3.main.c测试代码

我们在宋老师的例程lesson16_1中通过按红外遥控器上的按键得知,第一行的三个按键会使ircode[2]分别变为0x45、0x46、0x47显示在数码管上,那么我们只需要判断ircode[2]的数据就可以知道按下的是哪个按键然后去执行相关任务就可以了。

我们用小灯测试红外遥控器的按键是否有效,有效则会切换小灯的亮灭。

#include <reg52.h>
#include <function.h>//详见第六章第8讲
#include <infrared.h>
  
void main()
{
    LED_Init();//初始化LED硬件模块
    EA = 1; 
    InitInfrared();
   
    while (1)
    {
        if (irflag)  
        {
            irflag = 0;   
            //以下是遥控器第一行按键
            if(ircode[2]==0x45)LED9=!LED9;
            if(ircode[2]==0x46)LED8=!LED8;
            if(ircode[2]==0x47)LED7=!LED7;
       
            //以下是遥控器第二行按键
            if(ircode[2]==0x44)LED4=!LED4;
            if(ircode[2]==0x40)LED3=!LED3;
            if(ircode[2]==0x43)LED2=!LED2;    
        }  
    }
}

您可以捐助,支持我们的公益事业。

1元 10元 50元





认证码: 验证码,看不清楚?请点击刷新验证码 必填



498 次浏览
3次