STC学习:光照报警器

程序设计目标及程序运行效果说明
程序设计目标:学会利用AD采集光敏电阻的值,实现光照强度警报功能。
程序运行效果说明:key1改变光照警报器的模式
(1)当模式为警报下限时,通过用遮光板或者手指改变光敏电阻的光照强度,低于下限警报值时,蜂鸣器发声。
(2)按下Key1,当模式为警报上限时,通过用手电筒或闪光灯改变光敏电阻的光照强度,高于上限警报值时,蜂鸣器发声。

程序相关电路及工作原理说明
1.LED数码管电路
STC学习:光照报警器
2.光敏电阻电路
STC学习:光照报警器
3.工作原理
通过AD采集光敏电阻的值,检测AD求其平均值,这样可以达到稳定性。根据实际需要,设定警报上限和下限值,如果超过警报值,beep取反,产生方波,蜂鸣器发声。

测试方法
(1)用STC ISP默认设置,打开工程中的HEX并下载

(2)下载后观察现象为:当模式为警报下20时,通过用遮光板或者手指改变光敏电阻的光照强度,低于下限警报值时,蜂鸣器发声。按下Key1,当模式改为警报上限100时,通过用手电筒或闪光灯改变光敏电阻的光照强度,高于上限警报值时,蜂鸣器发声

代码如下:

#include "STC15F2K60S2.H"
sbit SEL0=P2^0;			//定义引脚
sbit SEL1=P2^1;
sbit SEL2=P2^2;
sbit SEL3=P2^3;
sbit beep=P3^4;		//蜂鸣器引脚
sbit Key1=P3^2;	   //Key1   切换标志位 
#define uint unsigned int
#define ulint unsigned long
#define uchar unsigned char	
#define NMAX_KEY 100  //100次读取按键值 
uint time=0;				//延时
uint l=0;						//执行光的次数
ulint suml=0;				//光AD值得总和
uint light=0;				//光
uint light_down=20; //警报下限
uint light_up=100;	//警报上限
uchar flag=0;				//切换模式,0代表下限,1代表上限
//设置用于显示光的三个变量
uint light_bai=0;		//百位
uint light_shi=0;		//十位
uint light_ge =0;		//个位
//警报值
uint down_bai=0;		//百位
uint down_shi=0;		//十位
uint down_ge =0;		//个位
//警报值
uint up_bai=0;		//百位
uint up_shi=0;		//十位
uint up_ge =0;		//个位
uchar G_count;
uchar display;
uint Key1_count;
uint Key2_count;
uint Key3_count;
uint Key_count;
bit Key1_C;				//key1当前的状态
bit Key1_P;				//key1前一个状态
//数码管上显示0-F
char segtable[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,
				 0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71};
void Delay(int n);
void weixuan(char i);
void SEG_Display();
void check();
void InitADC_light();
void date_processlight();
void date_processlight_down();
void date_processlight_up();
void Delay(int n)			//延时函数
{
	int x;
	while(n--)
	{
		x=60;
		while(x--);
	}
}
void weixuan(char i)	//数码管位的选择
{
	SEL2=i/4;
	SEL1=i%4/2;
	SEL0=i%2;
}
void SEG_Display()
{
	//用于设置光的位数
	P0=0;
	weixuan(5);
	P0=segtable[light_bai];
	Delay(10);
	P0=0;
	weixuan(6);
	P0=segtable[light_shi];
	Delay(10);
	P0=0;
	weixuan(7);
	P0=segtable[light_ge];
	Delay(10);
	if(flag==0)    //flag为0 显示下限值 020
	{
		P0=0;
		weixuan(0);
		P0=segtable[down_bai];
		Delay(10);
		P0=0;
		weixuan(1);
		P0=segtable[down_shi];
		Delay(10);
		P0=0;
		weixuan(2);
		P0=segtable[down_ge];
		Delay(10);
	} 
	if(flag==1)   //flag为1 显示上限值 100
	{
		P0=0;
		weixuan(0);
		P0=segtable[up_bai];
		Delay(10);
		P0=0;
		weixuan(1);
		P0=segtable[up_shi];
		Delay(10);
		P0=0;
		weixuan(2);
		P0=segtable[up_ge];
		Delay(10);
	}
}
void main()
{
	P1M1=0x00;            //将P1^7设置为推挽模式,其余为准双向口模式
	P1M0=0x80;         
	P0M1=0x00;						//设置P0为推挽模式,点亮数码管
	P0M0=0xff;
	P2M1=0x00;
	P2M0=0x08;						//将P2^3设置为推挽模式,其余为准双向口模式
	P3M1=0x00;
	P3M0=0x10;					  //设置P3^4为推挽模式
	SEL3=0;								//熄灭LED灯
	/*初始化所有按键的当前状态、前一个状态*/
	Key1_C=1;							//key1当前的状态
	Key1_P=1;							//key1前一个状态
	Key1_count=0x80+NMAX_KEY/3*2;
	Key_count=NMAX_KEY;
	IE=0xa8;							//EA=1打开总中断,EADC=1允许A/D转化中断,ET1=1允许T1中断	
	TMOD=0x10;						//使用定时器1,16位不可重装载模式,TH1、TL1全用
	TH1=(65535-100)/256;	//高8位赋初值,定时400周期
	TL1=(65535-100)%256;	//低8位赋初值
	TR1=1;								//启动定时器1
	InitADC_light();
	while(1)
		SEG_Display();
}
void InitADC_light()		//初始化光ADC
{
	P1ASF=0xff;           //将P1口作为模拟功能A/D使用
	ADC_RES=0;            //转换结果清零
	ADC_RESL=0;
	ADC_CONTR=0x8c;				//ADC_POWER=1打开A/D转换器电源;ADC_START=1启动模拟转换器ADC;CHS=100选择P1^4作为A/D输入使用
	CLK_DIV=0x20;
}
//分别取出温度和光照的百位、十位、个位
void date_processlight_down()
{
	down_bai=light_down%1000/100;
	down_shi=light_down%100/10;
	down_ge =light_down%10;
}
void date_processlight_up()
{
	up_bai=light_up%1000/100;
	up_shi=light_up%100/10;
	up_ge =light_up%10;
}
void date_processlight()
{
	light_bai=light%1000/100;
	light_shi=light%100/10;
	light_ge =light%10;
}
void check()
{
			if(Key1==0)
			   Key1_count--;
			Key_count--;					//总的次数减1
			if(Key_count==0)			//100次完了
			{
				if(Key1_count<0x80)
				{
					Key1_C=0;
					if(Key1_P==1)			//下降沿(按键做动作)
					{
						Key1_P=0;
						flag=!flag;
					}
				}
				if(Key1_count>=0x80)
				{
					Key1_C=1;
					if(Key1_P==0)
					   Key1_P=1;			//上升沿(假设不做动作那就继续)
				}
				//新一轮的判断
				Key1_count=0x80+NMAX_KEY/3*2;0
				Key_count=NMAX_KEY;
			}
}
void Timer1_Routine()interrupt 3	//3为定时器1中断编号
{
	IE=0x00;												//关闭总中断
	TR1=0;													//定时器1停止
	TH1=(65535-100)/256;						//重新赋值
	TL1=(65535-100)%256;
	check();
	if(flag==0)
	{
		if(light<light_down)
				beep=~beep;				  		 //产生方波使得蜂鸣器发声
		else beep=0;								 //不使用蜂鸣器时,讲引脚置0——处于低电平
	}
	if(flag==1)
	{
		if(light>light_up)
				beep=~beep;				 		  //产生方波使得蜂鸣器发声
		else beep=0;
	}
	IE=0xa8;											//打开总中断
	TR1=1;												//启动定时器1
}
//AD中断
void adc_isr()interrupt 5 using 1
{
    time++;
	IE=0x00;											//关闭中断
	if(time>2000)                 //取多次值求平均值减小误差
	{
		light=(suml+l/2)/l;				//四舍五入
		suml=0;
		l=0;
		time=0;
	   	date_processlight();
		date_processlight_down();
		date_processlight_up();
	}
	//处理光部分的数据
	l++;
	suml+=ADC_RES*256+ADC_RESL;//求l次AD值的和
	ADC_CONTR&=~0X10;  					 //转换完成后,ADC_FLAG清零
	ADC_CONTR|=0X08;	 					 //转换完成后,ADC_START赋1
	IE=0xa8;					 					 //打开中断
}
上一篇:2019.7.18 义乌模拟赛 T3 占领


下一篇:UWP 下拉刷新控件(PullToRefreshControl)