单片机C51-笔记-感应开关盖垃圾桶

一、项目描述

 二、编程描述

  • 开发步骤:
1. 舵机和超声波代码整合
       舵机用定时器 0
       超声波用定时器 1
       实现物体靠近后,自动开盖, 2 秒后关盖
2. 查询的方式添加按键控制
3. 查询的方式添加震动控制
4. 使用外部中断 0 配合震动控制

三、代码

1. 舵机和超声波代码整合
#include "reg52.h"

//距离小于10cm,D5亮,D6灭,反之相反现象

sbit D5       = P3^7;
sbit D6       = P3^6;
sbit Trig     = P1^5;
sbit Echo     = P1^6;
sbit sg90_con = P1^1;//像P1^1口输出一些东西

int jd;
int cnt = 0;

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

	//_nop_();
	//_nop_();
	i = 9;
	j = 104;
	k = 139;
	do
	{
		do
		{
			while (--k);
		} while (--j);
	} while (--i);
}


void Delay10us()		//@11.0592MHz
{
	unsigned char i;

	i =  2;
	while (--i);
}

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

	i = 18;
	j = 235;
	do
	{
		while (--j);
	} while (--i);
}

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

	//_nop_();
	//_nop_();
	i = 2;
	j = 67;
	k = 183;
	do
	{
		do
		{
			while (--k);
		} while (--j);
	} while (--i);
}


void Time0Init(){
	//1、配置定时器0工作模式位16位计时
	//TMOD = 0X01;
	TMOD &=0XF0;//设置定时器0的模式1
    TMOD |=0X01;
	//2、给初值,定一个0.5ms出来
	TL0 = 0X33;
	TH0 = 0XFE;
	//3、开始计时
	TR0 = 1;
	TF0 = 0;//赋初值
	//4、打开定时器0终端
	ET0 = 1;
	//5、打开总终端 
	EA = 1;
}

void Time1Init(){
	
	TMOD &=0X0F;//设置定时器1的模式1
    TMOD |=0X10;
	TH1 = 0;//从0开始计数
	TL1 = 0;
	//设置定时器0工作模式1,初始值设定0开始数数,不着急启动定时器
	
}

void startHC()
{
	
	Trig = 0;
	Trig = 1;
	Delay10us();
	Trig = 0;
	
}

double get_distance()
{
	
	double time;
	
	//定时器数据要清零
	TH1 = 0;//从0开始计数
	TL1 = 0;
	//1.给Trig端口一个10微妙的脉冲信号
	startHC();
    //2.Echo信号由低电平跳转到高电平,表示开始发送波
	while(Echo == 0);//当Echo一直是 低电平的时候,就是等着,等高电平不满足条件继续向下执行
	//波发出去的那一下开始启动定时器
	TR1 = 1;//定时器1
	//3.Echo信号由高电平跳转到低电平,表示开始波回来了
	while(Echo == 1);
	 //波回来那一下停止定时器
	TR1 = 0;
	//4.计算出中间经过多少时间
	/*
	十进制2左移1位,变成20,相当于乘以10
	二进制1左移1位,变成1 0(2),相当于乘以2,左移8位,乘以2的8次方=256;
	*/
	time = (TH1 * 256 + TL1)*1.085;//单位 微秒
	//5.计算距离=(速度(340m/s)*时间)/2
	//dis = 340m/s = 34000cm/s = 34cm/ms = 0.034cm/us/2=0.017
	 return (time * 0.017);
}

void openStatusLight()
{
	D5 = 0;
	Delay10ms();
	D6 = 1;
}
void closeStatusLight()
{
	D5 = 1;
	D6 = 0;
	Delay10ms();
}

void initSG90_0()
{
	jd = 1;//初始角度是0度,溢出一次就是0.5ms,高电平
	cnt = 0;//计数从0开始
	sg90_con = 1;//初始化舵机电平就为高电平
}

void openDusbin()
{
	//舵机开盖
	jd = 3;//0.5ms=0° 0.5*3 = 1.5ms = 95°
    cnt = 0;//每次改变角度让cnt从0开始
	Delay200ms();
}
void closeDusbin()
{
	//舵机关盖
	jd = 1;//0.5*1 = 0°
	cnt = 0;
	Delay200ms();
}

void main()
{
	
	double dis; 
	Time1Init();//定时器1初始化
	Time0Init();//定时器0初始化
	//舵机的初始位置 
	initSG90_0();
	
	while(1){
		//超声波测距
		dis = get_distance();
		if(dis <10){
			//开盖,灯状态,D5亮
			openStatusLight();
			openDusbin();
		}else{
			//关盖,灯状态,D5灭
			closeStatusLight();
			closeDusbin();
		}
		
	}	
 } 
 
  void Time0Handler() interrupt 1//定时器0的中中断查询序列是1
 {
	 cnt++;//统计爆表次数,cnt=1的时候,爆表了1次
	 //重新给初值
	 TL0 = 0X33;
	 TH0 = 0XFE;
	 if(cnt < jd){//画波形,让系统一上电就为高电平,当爆表一次之后,执行中断查询函数,在执行if语句的时候发现,
		         //cnt爆表小于一次,也就是刚经历的0.5ms让他为高电平,其他情况为低电平
		 sg90_con = 1;
	 }else{
		 sg90_con = 0;
	 } 
	 
	 if(cnt == 40){//爆表40次,总共经过了20ms
		 cnt = 0;//当100次表示1s,重新让cnt从1开始,计算下一次的1s
		 sg90_con = 1;//cnt一等于0就让它等于高电平
	 }
	 
 }
 
 

2. 查询的方式添加按键控制

只修改了两处

sbit SW1      = P2^1;//开关1

 if(dis <10 || SW1 == 0){//如果距离小于10厘米或者SW1按键被长按下,就开盖

#include "reg52.h"

//距离小于10cm,D5亮,D6灭,反之相反现象

sbit D5       = P3^7;
sbit D6       = P3^6;//灯
sbit SW1      = P2^1;//开关1
sbit Trig     = P1^5;
sbit Echo     = P1^6;
sbit sg90_con = P1^1;//像P1^1口输出一些东西

int jd;
int cnt = 0;

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

	//_nop_();
	//_nop_();
	i = 9;
	j = 104;
	k = 139;
	do
	{
		do
		{
			while (--k);
		} while (--j);
	} while (--i);
}


void Delay10us()		//@11.0592MHz
{
	unsigned char i;

	i =  2;
	while (--i);
}

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

	i = 18;
	j = 235;
	do
	{
		while (--j);
	} while (--i);
}

void Time0Init(){
	//1、配置定时器0工作模式位16位计时
	//TMOD = 0X01;
	TMOD &=0XF0;//设置定时器0的模式1
    TMOD |=0X01;
	//2、给初值,定一个0.5ms出来
	TL0 = 0X33;
	TH0 = 0XFE;
	//3、开始计时
	TR0 = 1;
	TF0 = 0;//赋初值
	//4、打开定时器0终端
	ET0 = 1;
	//5、打开总终端 
	EA = 1;
}

void Time1Init(){
	
	TMOD &=0X0F;//设置定时器1的模式1
    TMOD |=0X10;
	TH1 = 0;//从0开始计数
	TL1 = 0;
	//设置定时器0工作模式1,初始值设定0开始数数,不着急启动定时器
	
}

void startHC()
{
	
	Trig = 0;
	Trig = 1;
	Delay10us();
	Trig = 0;
	
}

double get_distance()
{
	
	double time;
	
	//定时器数据要清零
	TH1 = 0;//从0开始计数
	TL1 = 0;
	//1.给Trig端口一个10微妙的脉冲信号
	startHC();
    //2.Echo信号由低电平跳转到高电平,表示开始发送波
	while(Echo == 0);//当Echo一直是 低电平的时候,就是等着,等高电平不满足条件继续向下执行
	//波发出去的那一下开始启动定时器
	TR1 = 1;//定时器1
	//3.Echo信号由高电平跳转到低电平,表示开始波回来了
	while(Echo == 1);
	 //波回来那一下停止定时器
	TR1 = 0;
	//4.计算出中间经过多少时间
	/*
	十进制2左移1位,变成20,相当于乘以10
	二进制1左移1位,变成1 0(2),相当于乘以2,左移8位,乘以2的8次方=256;
	*/
	time = (TH1 * 256 + TL1)*1.085;//单位 微秒
	//5.计算距离=(速度(340m/s)*时间)/2
	//dis = 340m/s = 34000cm/s = 34cm/ms = 0.034cm/us/2=0.017
	 return (time * 0.017);
}

void openStatusLight()
{
	D5 = 0;
	Delay10ms();
	D6 = 1;
}
void closeStatusLight()
{
	D5 = 1;
	D6 = 0;
	
	Delay10ms();
}

void initSG90_0()
{
	jd = 1;//初始角度是0度,溢出一次就是0.5ms,高电平
	cnt = 0;//计数从0开始
	sg90_con = 1;//初始化舵机电平就为高电平
}

void openDusbin()
{
	//舵机开盖
	jd = 3;//0.5ms=0° 0.5*3 = 1.5ms = 95°
    cnt = 0;//每次改变角度让cnt从0开始

	Delay200ms();
}
void closeDusbin()
{
	//舵机关盖
	jd = 1;//0.5*1 = 0°
	cnt = 0;
	Delay200ms();
}

void main()
{
	
	double dis; 
	Time1Init();//定时器1初始化
	Time0Init();//定时器0初始化
	//舵机的初始位置 
	initSG90_0();
	
	while(1){
		//超声波测距
		dis = get_distance();
		if(dis <10 || SW1 == 0){//如果距离小于10厘米或者SW1按键被长按下,就开盖
			//开盖,灯状态,D5亮
			openStatusLight();
			openDusbin();
		}else{
			//关盖,灯状态,D5灭
			closeStatusLight();
			closeDusbin();
		}
		
	}	
 } 
 
  void Time0Handler() interrupt 1//定时器0的中中断查询序列是1
 {
	 cnt++;//统计爆表次数,cnt=1的时候,爆表了1次
	 //重新给初值
	 TL0 = 0X33;
	 TH0 = 0XFE;
	 if(cnt < jd){//画波形,让系统一上电就为高电平,当爆表一次之后,执行中断查询函数,在执行if语句的时候发现,
		         //cnt爆表小于一次,也就是刚经历的0.5ms让他为高电平,其他情况为低电平
		 sg90_con = 1;
	 }else{
		 sg90_con = 0;
	 } 
	 
	 if(cnt == 40){//爆表40次,总共经过了20ms
		 cnt = 0;//当100次表示1s,重新让cnt从1开始,计算下一次的1s
		 sg90_con = 1;//cnt一等于0就让它等于高电平
	 }
	 
 }
 
 

3. 查询的方式添加震动控制-使用外部中断0配合震动控制

#include "reg52.h"

//距离小于10cm,D5亮,D6灭,反之相反现象

sbit D5       = P3^7;
sbit D6       = P3^6;//灯
sbit SW1      = P2^1;//开关1
sbit Trig     = P1^5;
sbit Echo     = P1^6;
sbit sg90_con = P1^1;//像P1^1口输出一些东西
sbit vibrate  = P3^2;//这个接口可以请求外部中断

char jd;
char cnt = 0;
char mark_vibrate = 0;//标志位

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

	//_nop_();
	//_nop_();
	i = 9;
	j = 104;
	k = 139;
	do
	{
		do
		{
			while (--k);
		} while (--j);
	} while (--i);
}


void Delay10us()		//@11.0592MHz
{
	unsigned char i;

	i =  2;
	while (--i);
}

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

	i = 18;
	j = 235;
	do
	{
		while (--j);
	} while (--i);
}

void Time0Init(){
	//1、配置定时器0工作模式位16位计时
	//TMOD = 0X01;
	TMOD &=0XF0;//设置定时器0的模式1
    TMOD |=0X01;
	//2、给初值,定一个0.5ms出来
	TL0 = 0X33;
	TH0 = 0XFE;
	//3、开始计时
	TR0 = 1;
	TF0 = 0;//赋初值
	//4、打开定时器0终端
	ET0 = 1;
	//5、打开总终端 
	EA = 1;
}

void Time1Init(){
	
	TMOD &=0X0F;//设置定时器1的模式1
    TMOD |=0X10;
	TH1 = 0;//从0开始计数
	TL1 = 0;
	//设置定时器0工作模式1,初始值设定0开始数数,不着急启动定时器
	
}

void startHC()
{
	
	Trig = 0;
	Trig = 1;
	Delay10us();
	Trig = 0;
	
}

double get_distance()
{
	
	double time;
	
	//定时器数据要清零
	TH1 = 0;//从0开始计数
	TL1 = 0;
	//1.给Trig端口一个10微妙的脉冲信号
	startHC();
    //2.Echo信号由低电平跳转到高电平,表示开始发送波
	while(Echo == 0);//当Echo一直是 低电平的时候,就是等着,等高电平不满足条件继续向下执行
	//波发出去的那一下开始启动定时器
	TR1 = 1;//定时器1
	//3.Echo信号由高电平跳转到低电平,表示开始波回来了
	while(Echo == 1);
	 //波回来那一下停止定时器
	TR1 = 0;
	//4.计算出中间经过多少时间
	/*
	十进制2左移1位,变成20,相当于乘以10
	二进制1左移1位,变成1 0(2),相当于乘以2,左移8位,乘以2的8次方=256;
	*/
	time = (TH1 * 256 + TL1)*1.085;//单位 微秒
	//5.计算距离=(速度(340m/s)*时间)/2
	//dis = 340m/s = 34000cm/s = 34cm/ms = 0.034cm/us/2=0.017
	 return (time * 0.017);
}

void openStatusLight()
{
	D5 = 0;
	Delay10ms();
	D6 = 1;
}
void closeStatusLight()
{
	D5 = 1;
	D6 = 0;
	
	Delay10ms();
}

void initSG90_0()
{
	jd = 1;//初始角度是0度,溢出一次就是0.5ms,高电平
	cnt = 0;//计数从0开始
	sg90_con = 1;//初始化舵机电平就为高电平
}

void openDusbin()
{
	//舵机开盖
	jd = 3;//0.5ms=0° 0.5*3 = 1.5ms = 95°
    cnt = 0;//每次改变角度让cnt从0开始

	Delay200ms();
}
void closeDusbin()
{
	//舵机关盖
	jd = 1;//0.5*1 = 0°
	cnt = 0;
	Delay200ms();
}
void EX0_Init()
{
	//打开外部中断
	EX0 = 1;
	//低电平触发
	IT0 = 0;
}

void main()
{
	
	double dis; 
	Time1Init();//定时器1初始化
	Time0Init();//定时器0初始化
	EX0_Init();//外部中断
	
	//舵机的初始位置 
	initSG90_0();
	
	while(1){
		//超声波测距
		dis = get_distance();
		if(dis <10 || SW1 == 0 || mark_vibrate == 1){//如果距离小于10厘米或者SW1按键被按下或者硬件感觉到震动,启动中断触发 ,就开盖
			//开盖,灯状态,D5亮
			openStatusLight();
			openDusbin();
			mark_vibrate = 0; //手动代码帮忙恢复
		}else{
			//关盖,灯状态,D5灭
			closeStatusLight();
			closeDusbin();
		}
		
	}	
 } 
 
  void Time0Handler() interrupt 1//定时器0的中中断查询序列是1
 {
	 cnt++;//统计爆表次数,cnt=1的时候,爆表了1次
	 //重新给初值
	 TL0 = 0X33;
	 TH0 = 0XFE;
	 if(cnt < jd){//画波形,让系统一上电就为高电平,当爆表一次之后,执行中断查询函数,在执行if语句的时候发现,
		         //cnt爆表小于一次,也就是刚经历的0.5ms让他为高电平,其他情况为低电平
		 sg90_con = 1;
	 }else{
		 sg90_con = 0;
	 } 
	 
	 if(cnt == 40){//爆表40次,总共经过了20ms
		 cnt = 0;//当100次表示1s,重新让cnt从1开始,计算下一次的1s
		 sg90_con = 1;//cnt一等于0就让它等于高电平
	 }
	 
 }
void EX0_Handler() interrupt 0
{
	mark_vibrate = 1;
	 
}
 
 

4、震动、测距、按键开关垃圾桶盖,并且在开马桶盖之前响滴的一声,解决马桶盖抽抽的问题

代码如下:

#include "reg52.h"

//距离小于10cm,D5亮,D6灭,反之相反现象

sbit D5       = P3^7;
sbit D6       = P3^6;//灯
sbit SW1      = P2^1;//开关1
sbit Trig     = P1^5;
sbit Echo     = P1^6;
sbit sg90_con = P1^1;//像P1^1口输出一些东西
sbit vibrate  = P3^2;//这个接口可以请求外部中断
sbit beep     = P2^0;//蜂鸣器

char jd;
char jd_bak;//标志位
char cnt = 0;
char mark_vibrate = 0;//标志位

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

	//_nop_();
	//_nop_();
	i = 9;
	j = 104;
	k = 139;
	do
	{
		do
		{
			while (--k);
		} while (--j);
	} while (--i);
}


void Delay10us()		//@11.0592MHz
{
	unsigned char i;

	i =  2;
	while (--i);
}

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

	i = 18;
	j = 235;
	do
	{
		while (--j);
	} while (--i);
}

void Time0Init(){
	//1、配置定时器0工作模式位16位计时
	//TMOD = 0X01;
	TMOD &=0XF0;//设置定时器0的模式1
    TMOD |=0X01;
	//2、给初值,定一个0.5ms出来
	TL0 = 0X33;
	TH0 = 0XFE;
	//3、开始计时
	TR0 = 1;
	TF0 = 0;//赋初值
	//4、打开定时器0终端
	ET0 = 1;
	//5、打开总终端 
	EA = 1;
}

void Time1Init(){
	
	TMOD &=0X0F;//设置定时器1的模式1
    TMOD |=0X10;
	TH1 = 0;//从0开始计数
	TL1 = 0;
	//设置定时器0工作模式1,初始值设定0开始数数,不着急启动定时器
	
}

void startHC()
{
	
	Trig = 0;
	Trig = 1;
	Delay10us();
	Trig = 0;
	
}

double get_distance()
{
	
	double time;
	
	//定时器数据要清零
	TH1 = 0;//从0开始计数
	TL1 = 0;
	//1.给Trig端口一个10微妙的脉冲信号
	startHC();
    //2.Echo信号由低电平跳转到高电平,表示开始发送波
	while(Echo == 0);//当Echo一直是 低电平的时候,就是等着,等高电平不满足条件继续向下执行
	//波发出去的那一下开始启动定时器
	TR1 = 1;//定时器1
	//3.Echo信号由高电平跳转到低电平,表示开始波回来了
	while(Echo == 1);
	 //波回来那一下停止定时器
	TR1 = 0;
	//4.计算出中间经过多少时间
	/*
	十进制2左移1位,变成20,相当于乘以10
	二进制1左移1位,变成1 0(2),相当于乘以2,左移8位,乘以2的8次方=256;
	*/
	time = (TH1 * 256 + TL1)*1.085;//单位 微秒
	//5.计算距离=(速度(340m/s)*时间)/2
	//dis = 340m/s = 34000cm/s = 34cm/ms = 0.034cm/us/2=0.017
	 return (time * 0.017);
}

void openStatusLight()
{
	D5 = 0;
	Delay10ms();
	D6 = 1;
}
void closeStatusLight()
{
	D5 = 1;
	D6 = 0;
	
	Delay10ms();
}

void initSG90_0()
{
	jd = 1;//初始角度是0度,溢出一次就是0.5ms,高电平
	cnt = 0;//计数从0开始
	sg90_con = 1;//初始化舵机电平就为高电平
}

void openDusbin()
{
	char n;
	//舵机开盖
	jd = 3;//0.5ms=0° 0.5*3 = 1.5ms = 95°
	if(jd_bak != jd){
		cnt = 0;//每次改变角度让cnt从0开始
	    beep = 0;//响 
	    for(n=0;n<20;n++)
	      Delay10ms();
	    beep = 1;//不响
	    Delay200ms();
	}
	jd_bak = jd;
	
    
}
void closeDusbin()
{
	//舵机关盖
	jd = 1;//0.5*1 = 0°
	jd_bak = jd;
	cnt = 0; 
	Delay200ms();
}
void EX0_Init()
{
	//打开外部中断
	EX0 = 1;
	//低电平触发
	IT0 = 0;
}

void main()
{
	
	double dis; 
	Time1Init();//定时器1初始化
	Time0Init();//定时器0初始化
	EX0_Init();//外部中断
	
	//舵机的初始位置 
	initSG90_0();
	
	while(1){
		//超声波测距
		dis = get_distance();
		if(dis <10 || SW1 == 0 || mark_vibrate == 1){//如果距离小于10厘米或者SW1按键被按下或者硬件感觉到震动,启动中断触发 ,就开盖
			//开盖,灯状态,D5亮
			openStatusLight(); 
			openDusbin();
			mark_vibrate = 0; //手动代码帮忙恢复
		}else{
			//关盖,灯状态,D5灭
			closeStatusLight();
			closeDusbin();
		}
		
	}	
 } 
 
  void Time0Handler() interrupt 1//定时器0的中中断查询序列是1
 {
	 cnt++;//统计爆表次数,cnt=1的时候,爆表了1次
	 //重新给初值
	 TL0 = 0X33;
	 TH0 = 0XFE;
	 if(cnt < jd){//画波形,让系统一上电就为高电平,当爆表一次之后,执行中断查询函数,在执行if语句的时候发现,
		         //cnt爆表小于一次,也就是刚经历的0.5ms让他为高电平,其他情况为低电平
		 sg90_con = 1;
	 }else{
		 sg90_con = 0;
	 } 
	 
	 if(cnt == 40){//爆表40次,总共经过了20ms
		 cnt = 0;//当100次表示1s,重新让cnt从1开始,计算下一次的1s
		 sg90_con = 1;//cnt一等于0就让它等于高电平
	 }
	 
 }
void EX0_Handler() interrupt 0
{
	mark_vibrate = 1;
	 
}
 
 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值