C51---智能感应垃圾桶

1.器件:C51、蜂鸣器、SG90舵机、HC-SR04超声波测距

2.实现功能:超声波感应,舵机转动打开垃圾桶,蜂鸣器发出提示,LED灯亮

3.代码:

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

//距离小于10cm,D5亮,D6灭,反之相反现象
sbit D5        = P3^7;//根据原理图(电路图),设备变量led1指向P3组IO口的第七口
sbit D6        = P3^6;//根据原理图(电路图),设备变量led2指向P3组IO口的第六口
sbit Trig      = P1^5;
sbit Echo      = P1^6;
sbit sg90_con  = P1^1;    //定义sg90的管脚
sbit beep             = P2^0;    

int  jiaodu;                    //定义一个舵机转动的角度
int  cnt = 0;                    //cnt要定为全局变量

void Delay150ms()        //@11.0592MHz
{
    unsigned char i, j, k;

    _nop_();
    i = 2;
    j = 13;
    k = 237;
    do
    {
        do
        {
            while (--k);
        } while (--j);
    } while (--i);
}

void Delay2000ms()        //@11.0592MHz
{
    unsigned char i, j, k;

    //_nop_();
    i = 15;
    j = 2;
    k = 235;
    do
    {
        do
        {
            while (--k);
        } while (--j);
    } while (--i);
}

void Delay10us()        //@11.0592MHz
{
    unsigned char i;
    i = 2;
    while (--i);
}

//定时器 0 初始化
void Time0Init()
{
//1.配置定时器0工作模式位16位计时
    TMOD &= 0xF0;        //设置定时器模式
    TMOD |= 0x01;        //设置定时器模式
//2.给初值,定一个10ms出来
    TL0 = 0x33;        //设置定时初值
    TH0 = 0xFE;        //设置定时初值
//3.开始计时
    TR0 = 1;
    TF0 = 0;
//4.打开定时器中断
    ET0 = 1; 
//5.打开总中断EA
    EA = 1;
}

//定时器1初始化
void Time1Init()
{
    TMOD &= 0x0F;        //设置定时器模式
    TMOD |= 0x10;        //设置定时器模式
    TH0 = 0;
    TL0 = 0;
    //设置定时器0工作模式,初始值设定0开始数数,不着急启动定时器 
}


void startHC()
{
    Trig = 0;
    Trig = 1;
    Delay10us();
    Trig = 0;
    
}
 
double get_distance()
{
    double time;
    //定时器数据要清零,以便下一次测距
        TH1 = 0;
        TL1 = 0;
    //1.Trig,给Trig端口至少10us的高电平
    startHC();
    //2.由低电平跳转到高电平,表示开始发送波; 波发出去那一下,开始启动定时器
    while(Echo == 0);
    TR1 = 1;                     
    //3.由高电平跳转回低电平,表示波回来了波回来那一下我们开始停止定时器
    while(Echo == 1);
    TR1 = 0;     
    //4.计算出中间经过多少时间
    time = (TH1 * 256 + TL1)*1.085;//us为单位
    //5.距离 = 速度(340m/s)*时间/2        dis = 340m/s == 3400cm/s == 34cm/ms == 0.034cm/us
    return (time * 0.017);

}
//封装开灯状态
void openStatusLight()
{
            D5 = 0;    
          D6 = 1;
}
//封装关灯状态
void closeStatusLight()
{
            D5 = 1;
          D6 = 0;
}

void initSG90_jiaodu()
{
    jiaodu = 4;//初始化角度0度,0.5ms,高电平
    cnt   = 0;
    sg90_con = 1;//一开始从高电平开始
}

void openDusbin()
{
            char n;
            //舵机开盖
                jiaodu = 3;    //90度,1.5ms高电平
                   cnt = 0;
                  beep = 0;
        for(n=0;n<2;n++)
            Delay150ms();
                    beep = 1; 
          Delay2000ms();
}

void closeDusbin()
{
                //舵机关盖
                jiaodu = 1;    //0度,1.5ms高电平
                     cnt = 0;
            Delay150ms();
}

void main()
{
    
    double  dis;    
    Time0Init();    
    Time1Init();
    
    //舵机回到初始位置
    initSG90_jiaodu();
    
    while(1){
        //超声波测距
     dis = get_distance();
     if(dis <10){
                //开盖,D5亮 
                openStatusLight();//调用开灯状态
                openDusbin();            //舵机开盖
        }else {
                //关盖,D5灭
                closeStatusLight();//调用关灯状态
                closeDusbin();         //舵机关盖
        } 
    }
}

//封装一个中断函数
void Time0Handler() interrupt 1 
{
        cnt++;//统计爆表次数,cnt等于1的时候,爆表了一次
        //重新给初值
        TL0 = 0x33;        
        TH0 = 0xFE;    
        sg90_con = 1;
    
    //控制PWM波
        if(cnt < jiaodu){  //cnt 决定舵机转动的角度
                    sg90_con = 1;
        }else{
                    sg90_con = 0;
            }
    
        if(cnt == 40){//cnt爆表了40次,经过了20ms
            cnt = 0;//当经过了1s,重新让cnt从0开始,计算下一次的1s.
                    sg90_con = 1;
    }
}

4.效果图

 

  • 3
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值