51单片机——感应开关盖垃圾桶

51单片机------感应开关盖垃圾桶

1.定时器点灯

c 复制代码
#include "reg52.h"

sbit led = P3^6;

void main()
{
	int cnt = 0;
	led = 1;
	
	//1. 配置定时器0工作模式位16位计时
	TMOD = 0x01;
	//2. 给初值,定一个10ms出来
	TL0=0x00;
	TH0=0xDC;
	//3. 开始计时
	TR0 = 1;
	TF0 = 0;
	
	while(1){
		if(TF0 == 1)//当爆表的时候,硬件会修改bit5(TF0)位上面的数据,改成1
		{
			TF0 = 0;//不用中断,必须软件清零
			cnt++;  //统计爆表的次数
			//重新给初值
			TL0=0x00;
			TH0=0xDC;
			if(cnt == 100){//爆表100次,经过了1s
				cnt = 0;  //当100次表示1s,重新让cnt从0开始,计算下一次的1s
				led = !led;//每经过1s,翻转led的状态
			}
		}
	}
}
c 复制代码
#include "reg52.h"

sbit led = P3^6;

void delay10ms()
{
	//1. 配置定时器0工作模式位16位计时
	TMOD = 0x01;
	//2. 给初值,定一个10ms出来
	TL0=0x00;
	TH0=0xDC;
	//3. 开始计时
	TR0 = 1;
	TF0 = 0;
}


void main()
{
	int cnt = 0;
	led = 1;
	
	delay10ms();
	
	while(1){
		if(TF0 == 1)//当爆表的时候,硬件会修改bit5(TF0)位上面的数据,改成1
		{
			TF0 = 0;//不用中断,必须软件清零
			cnt++;  //统计爆表的次数
			//重新给初值
			TL0=0x00;
			TH0=0xDC;
			if(cnt == 100){//爆表100次,经过了1s
				cnt = 0;  //当100次表示1s,重新让cnt从0开始,计算下一次的1s
				led = !led;//每经过1s,翻转led的状态
			}
		}
	}
}

2.定时器中断点灯

c 复制代码
#include "reg52.h"

sbit led = P3^6;
sbit led1 = P3^7;
int cnt = 0;

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

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

	i = 3;
	j = 26;
	k = 223;
	do
	{
		do
		{
			while (--k);
		} while (--j);
	} while (--i);
}

void main()
{
	led = 1;
	Time0Init();
	while(1){
		led1 = 0;
		Delay300ms();
		led1 = 1;
		Delay300ms();
	}
}

void Time0Handler() interrupt 1
{
	cnt++;  //统计爆表的次数
	//重新给初值
	TL0=0x00;
	TH0=0xDC;
	if(cnt == 100){//爆表100次,经过了1s
		cnt = 0;  //当100次表示1s,重新让cnt从0开始,计算下一次的1s
		led = !led;//每经过1s,翻转led的状态
	}
		
}

3.PWM开发SG90舵机

c 复制代码
#include "reg52.h"

sbit sg90_con = P1^1;
int jd;
int cnt = 0;


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

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


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

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

	i = 3;
	j = 26;
	k = 223;
	do
	{
		do
		{
			while (--k);
		} while (--j);
	} while (--i);
}

void main()
{
	Delay300ms();//让硬件稳定一下
	Time0Init(); //初始化定时器
	jd = 1;      //初始角度是0度,0.5ms,溢出1就是0.5,高电平
	cnt = 0;
	sg90_con = 1;//一开始从高电平开始
	
	//每隔两秒切换一次角度
	while(1){
		jd = 3; //90度 1.5ms高电平
		cnt = 0;
		Delay2000ms();
		jd = 1; //0度
		cnt = 0;
		Delay2000ms();
	}
}

void Time0Handler() interrupt 1
{
	cnt++;  //统计爆表的次数. cnt=1的时候,报表了1
	//重新给初值
	TL0=0x33;
	TH0=0xFE;
	
	//控制PWM波
	if(cnt < jd){
		sg90_con = 1;
	}else{
		sg90_con = 0;
	}
	
	if(cnt == 40){//爆表40次,经过了20ms
		cnt = 0;  //当100次表示1s,重新让cnt从0开始,计算下一次的1s
		sg90_con = 1;
	}
		
}

4.超声波测距

c 复制代码
#include "reg52.h"

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

sbit D5 = P3^7;//根据原理图(电路图),设备变量led1指向P3组IO口的第7口
sbit D6 = P3^6;//根据原理图(电路图),设备变量led2指向P3组IO口的第6口
sbit Trig = P1^5;
sbit Echo = P1^6;

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

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

void Time0Init()
{	
	TMOD &= 0xF0;		//设置定时器模式
	TMOD |= 0x01;
	TH0 = 0;
	TL0 = 0;
	//设置定时器0工作模式1,初始值设定0开始数数,不着急启动定时器
}
/*
十进制2左移1位,变成20。相当于乘以10
二禁止1左移1位,变成10(2)。相当于乘以2,左移8位,乘以2的8次方=256;*/

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

项目开发-感应开关盖垃圾桶

1.超声波封装

c 复制代码
#include "reg52.h"

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

sbit D5 = P3^7;//根据原理图(电路图),设备变量led1指向P3组IO口的第7口
sbit D6 = P3^6;//根据原理图(电路图),设备变量led2指向P3组IO口的第6口
sbit Trig = P1^5;
sbit Echo = P1^6;

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

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

void Time1Init()
{	
	TMOD &= 0x0F;		//设置定时器模式
	TMOD |= 0x10;
	TH1 = 0;
	TL1 = 0;
	//设置定时器0工作模式1,初始值设定0开始数数,不着急启动定时器
}
/*
十进制2左移1位,变成20。相当于乘以10
二禁止1左移1位,变成10(2)。相当于乘以2,左移8位,乘以2的8次方=256;*/

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

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

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

void main()
{
	
	double dis;
	
	Time1Init();
	
	while(1){
		
		dis = get_distance();
		if(dis < 10){
				openStatusLight();
		}else{
				closeStatusLight();
		}
	
	}
}

2.测距开关盖

c 复制代码
#include "reg52.h"

sbit D5       = P3^7;//根据原理图(电路图),设备变量led1指向P3组IO口的第7口
sbit D6       = P3^6;//根据原理图(电路图),设备变量led2指向P3组IO口的第6口
sbit Trig     = P1^5;
sbit Echo     = P1^6;
sbit sg90_con = P1^1;

int jd;
int cnt = 0;

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

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


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

	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);
}

void Time0Init()
{
	//1. 配置定时器0工作模式位16位计时
	TMOD &= 0xF0;		//设置定时器模式
	TMOD |= 0x01;
	//2. 给初值,定一个0.5出来
	TL0=0x33;
	TH0=0xFE;
	//3. 开始计时
	TR0 = 1;
	TF0 = 0;
	//4. 打开定时器0中断
	ET0 = 1;
	//5. 打开总中断EA
	EA = 1;
}
void Time1Init()
{	
	TMOD &= 0x0F;		//设置定时器模式
	TMOD |= 0x10;
	TH1 = 0;
	TL1 = 0;
	//设置定时器0工作模式1,初始值设定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. echo由低电平跳转到高电平,表示开始发送波
		while(Echo == 0);
		//波发出去的那一下,开始启动定时器
		TR1 = 1;
		//3. 由高电平跳转回低电平,表示波回来了
		while(Echo == 1);
		//波回来的那一下,我们开始停止定时器
		TR1 = 0;
		//4. 计算出中间经过多少时间
		time = (TH1 * 256 + TL1)*1.085;//us为单位
		//5. 距离 = 速度 (340m/s)* 时间/2
		return  (time * 0.017);
}

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

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

void openDusbin()
{
	//舵机开盖
	jd = 3; //90度 1.5ms高电平
	cnt = 0;
	Delay2000ms();
}

void closeDusbin()
{
	//关盖
	jd = 1; //0度
	cnt = 0;
	Delay150ms();
}
void main()
{
	
	double dis;
	
	Time0Init();
	Time1Init();
	//舵机的初始位置
	initSG90_0();
	
	while(1){
		//超声波测距
		dis = get_distance();
		if(dis < 10){//如果小于10厘米
				//开盖,灯状态,D5亮
				openStatusLight();
				openDusbin();
		}else{
				//关盖,灯状态,D5灭
				closeStatusLight();
				closeDusbin();
		}
	}
}

void Time0Handler() interrupt 1
{
	cnt++;  //统计爆表的次数. cnt=1的时候,报表了1
	//重新给初值
	TL0=0x33;
	TH0=0xFE;
	
	//控制PWM波
	if(cnt < jd){
		sg90_con = 1;
	}else{
		sg90_con = 0;
	}
	
	if(cnt == 40){//爆表40次,经过了20ms
		cnt = 0;  //当100次表示1s,重新让cnt从0开始,计算下一次的1s
		sg90_con = 1;
	}
		
}

3.添加震动开关盖和按键开关盖

c 复制代码
#include "reg52.h"

sbit D5       = P3^7;//根据原理图(电路图),设备变量led1指向P3组IO口的第7口
sbit D6       = P3^6;//根据原理图(电路图),设备变量led2指向P3组IO口的第6口
sbit SW1      = P2^1;
sbit Trig     = P1^5;
sbit Echo     = P1^6;
sbit sg90_con = P1^1;
sbit vibrate  = P3^2;

char jd;
char cnt = 0;
char mark_vibrate = 0;

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

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


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

	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);
}

void Time0Init()
{
	//1. 配置定时器0工作模式位16位计时
	TMOD &= 0xF0;		//设置定时器模式
	TMOD |= 0x01;
	//2. 给初值,定一个0.5出来
	TL0=0x33;
	TH0=0xFE;
	//3. 开始计时
	TR0 = 1;
	TF0 = 0;
	//4. 打开定时器0中断
	ET0 = 1;
	//5. 打开总中断EA
	EA = 1;
}
void Time1Init()
{	
	TMOD &= 0x0F;		//设置定时器模式
	TMOD |= 0x10;
	TH1 = 0;
	TL1 = 0;
	//设置定时器0工作模式1,初始值设定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. echo由低电平跳转到高电平,表示开始发送波
		while(Echo == 0);
		//波发出去的那一下,开始启动定时器
		TR1 = 1;
		//3. 由高电平跳转回低电平,表示波回来了
		while(Echo == 1);
		//波回来的那一下,我们开始停止定时器
		TR1 = 0;
		//4. 计算出中间经过多少时间
		time = (TH1 * 256 + TL1)*1.085;//us为单位
		//5. 距离 = 速度 (340m/s)* 时间/2
		return  (time * 0.017);
}

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

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

void openDusbin()
{
	//舵机开盖
	jd = 3; //90度 1.5ms高电平
	cnt = 0;
	Delay2000ms();
}

void closeDusbin()
{
	//关盖
	jd = 1; //0度
	cnt = 0;
	Delay150ms();
}

void EX0_Init()
{
	//打开外部中断
	EX0 = 1;
	//低电平触发
	IT0 = 0;
}
void main()
{
	
	double dis;
	
	Time0Init();
	Time1Init();
	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
{
	cnt++;  //统计爆表的次数. cnt=1的时候,报表了1
	//重新给初值
	TL0=0x33;
	TH0=0xFE;
	
	//控制PWM波
	if(cnt < jd){
		sg90_con = 1;
	}else{
		sg90_con = 0;
	}
	
	if(cnt == 40){//爆表40次,经过了20ms
		cnt = 0;  //当100次表示1s,重新让cnt从0开始,计算下一次的1s
		sg90_con = 1;
	}
		
}

void Ex0_Handler() interrupt 0
{
	mark_vibrate = 1;
}

4.添加蜂鸣器-项目完结

功能描述:

c 复制代码
检测靠近时,垃圾桶自动开盖并伴随滴一声,2秒后关盖
发生震动时,垃圾桶自动开盖并伴随滴一声,2秒后关盖
按下按键时,垃圾桶自动开盖并伴随滴一声,2秒后关盖
c 复制代码
#include "reg52.h"

sbit D5       = P3^7;//根据原理图(电路图),设备变量led1指向P3组IO口的第7口
sbit D6       = P3^6;//根据原理图(电路图),设备变量led2指向P3组IO口的第6口
sbit SW1      = P2^1;
sbit Trig     = P1^5;
sbit Echo     = P1^6;
sbit sg90_con = P1^1;
sbit vibrate  = P3^2;
sbit beep     = P2^0;

char jd;
char jd_bak;
char cnt = 0;
char mark_vibrate = 0;

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

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


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

	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);
}

void Time0Init()
{
	//1. 配置定时器0工作模式位16位计时
	TMOD &= 0xF0;		//设置定时器模式
	TMOD |= 0x01;
	//2. 给初值,定一个0.5出来
	TL0=0x33;
	TH0=0xFE;
	//3. 开始计时
	TR0 = 1;
	TF0 = 0;
	//4. 打开定时器0中断
	ET0 = 1;
	//5. 打开总中断EA
	EA = 1;
}
void Time1Init()
{	
	TMOD &= 0x0F;		//设置定时器模式
	TMOD |= 0x10;
	TH1 = 0;
	TL1 = 0;
	//设置定时器0工作模式1,初始值设定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. echo由低电平跳转到高电平,表示开始发送波
		while(Echo == 0);
		//波发出去的那一下,开始启动定时器
		TR1 = 1;
		//3. 由高电平跳转回低电平,表示波回来了
		while(Echo == 1);
		//波回来的那一下,我们开始停止定时器
		TR1 = 0;
		//4. 计算出中间经过多少时间
		time = (TH1 * 256 + TL1)*1.085;//us为单位
		//5. 距离 = 速度 (340m/s)* 时间/2
		return  (time * 0.017);
}

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

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

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

void closeDusbin()
{
	//关盖
	jd = 1; //0度
	jd_bak = jd;
	cnt = 0;
	Delay150ms();
}

void EX0_Init()
{
	//打开外部中断
	EX0 = 1;
	//低电平触发
	IT0 = 0;
}
void main()
{
	
	double dis;
	
	Time0Init();
	Time1Init();
	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
{
	cnt++;  //统计爆表的次数. cnt=1的时候,报表了1
	//重新给初值
	TL0=0x33;
	TH0=0xFE;
	
	//控制PWM波
	if(cnt < jd){
		sg90_con = 1;
	}else{
		sg90_con = 0;
	}
	
	if(cnt == 40){//爆表40次,经过了20ms
		cnt = 0;  //当100次表示1s,重新让cnt从0开始,计算下一次的1s
		sg90_con = 1;
	}
		
}

void Ex0_Handler() interrupt 0
{
	mark_vibrate = 1;
}
相关推荐
gscsded2 小时前
C2000 GPIO 配置笔记
单片机
Sakuyu434682 小时前
STM32基础
stm32·单片机·嵌入式硬件
gscsded2 小时前
C2000 CPU Timer 学习笔记
单片机
iCxhust3 小时前
AD0808调试笔记
笔记·单片机·嵌入式硬件·操作系统·微机原理·8088单板机
木子单片机4 小时前
基于51单片机的步进电机调速系统设计
单片机·嵌入式硬件·51单片机·keil
三易串口屏5 小时前
实验10 物理按键实验
单片机·51单片机·串口屏·串口协议·uart 通信·嵌入式 ui
深圳光特通信豆子5 小时前
10Mb/s TTL单模双纤光收发模块技术规格手册
单片机·嵌入式硬件
一路往蓝-Anbo5 小时前
第六章:RTOS 任务 —— 任务逻辑与并发的 TDD 路径
网络·stm32·单片机·嵌入式硬件·tdd
星夜夏空995 小时前
STM32单片机学习(20) —— 利用中断实现串口通信(填前面的坑)
stm32·单片机·学习
wengqidaifeng5 小时前
2026年电赛校赛备战MSPM0G3507+keil讲解(上)-----2025年电赛E题小车篇
单片机·嵌入式硬件·电赛