基于51单片机的温度报警控制系统Protues仿真设计

目录

一、设计背景

二、实现功能

三、总体硬件设计

四、仿真演示

四、源程序

一、设计背景

        随着现代工农业技术的发展及人们对生活环境要求的提高,人们也迫切需要检测与了解环境温度。特别地,高温情况下极易造成火灾,例如,在机房中,电脑等设备发热快,若在短时间内机房温度升高,超出了设备正常温度,就会导致系统瘫痪或产生火灾。低温可能导致设备的处理器运行速度变慢,RAM(随机访问存储器)响应时间增加,以及设备的整体性能下降。特别是对于电子组件和机械设备,低温会使其反应速度减缓。在这种情况下,温度如果得到及时的控制与报警,就可以在一定程度上避免上述情况的发生。  

       另一个方面,单片机技术已经普及到各个领域,已经成为一种比较成熟的技术。因此本文将利用51单片机芯片,设计一种基于51单片机的温度报警控制系统。通过仿真实验证明,本文设计的温度报警控制系统工作时,液晶显示屏实时显示温度值并利用按键设定温度上下限范围,当所测温度超过设定的温度范围时,蜂鸣器发出报警信号,启动风扇降温;当所测温度低于设定的温度范围时,蜂鸣器发出报警信号,启动加热器进行加热。本设计实现的温控系统结构简单,可操作性强,具有一定的实用价值。

二、实现功能

        本设计以51单片机为控制核心,设计一种温度报警控制系统。可具体实现以下功能:

     (1)系统通过DS18B20采集当前温度环境温度并显示(精确到0.1度);

     (2)系统包含三个按键,可设定上下限温度的阈值;

     (3)若当前温度高于上限阈值温度时,声光报警,开启电机降温;若当前温度低于下限阈值温度时,声光报警,启动加热装置。

三、总体硬件设计

       本系统设计的基于51单片机的温度报警控制系统,总体设计框图如图1所示,主要包括单片机最小系统、温度采集电路、独立式按键电路、显示电路、报警电路、加热电路和散热电路,其中主控芯片采用功耗低、性能高的单片机 STC89C52,温度采集电路采用数字温度传感器DS18B20,显示电路采用 LCD1602 液晶显示器,报警电路采用蜂鸣器以及LED 指示灯设计实现声光报警,独立式按键用来设置不同时间段温度报警的上下限阈值。当实测环境温度值大于的温度上限值时,系统自动进入散热模式,直流电机运转带动风扇工作,同时蜂鸣器响、红色LED 指示灯点亮 ;若低于设定的温度下限阈值,系统自动进入加热模式,继电器控制加热设备工作,同时蜂鸣器响、绿色LED 指示灯点亮 ;若当前温度处于设定的温度上下限阈值之间时,关闭散热、加热及报警,从而使温度控制在设定的范围内。

          

                                                         图1 系统总体设计框图

四、仿真演示

当实测环境温度值处于设定的温度上下限阈值之间时,关闭散热、加热及报警。

当实测环境温度值大于设定的温度上限阈值时,直流电机运转带动风扇工作,系统进入散热模式,同时蜂鸣器响、红色LED 指示灯点亮。

当实测环境温度值低于设定的温度下限阈值,系统进入加热模式,继电器控制加热设备工作,同时蜂鸣器响、绿色LED 指示灯点亮 。

通过三个按键,设定上下限温度阈值,设置过程中LCD上会有光标提示。

四、源程序

#include <reg51.h> 
#define uint unsigned int
#define uchar unsigned char   	//宏定义
#define LCD1602 P0

sbit SET=P3^1;    			//定义调整键
sbit DEC=P3^2;    			//定义减少键
sbit ADD=P3^3;    			//定义增加键
sbit BUZZ=P3^6;    			//定义蜂鸣器
sbit ALAM=P1^2;				//定义灯光报警
sbit fengshan=P1^0;   //定义风扇降温
sbit jiare=P1^1;      //定义加热
sbit ALAM1=P1^4;
sbit DQ=P3^7;     			//定义DS18B20总线I/O	
sbit RS = P2^7;
sbit EN = P2^6;
bit shanshuo_st;    			//闪烁间隔标志
bit beep_st;     				//蜂鸣器间隔标志
uchar x=0;      				//计数器

uchar code tab1[]={" Temper:   .  C "};
uchar code tab2[]={"TH:   C  TL:   C"};
uint c;
uchar Mode=0;     			//状态标志
signed char TH=30;  		//上限报警温度,默认值为30
signed char TL=20;   		//下限报警温度,默认值为20
//============================================================================================
//====================================DS18B20=================================================
//============================================================================================
/*****延时子程序*****/
void Delay_DS18B20(int num)
{
  while(num--) ;
}
void delay(uint xms)//延时函数,有参函数
{
	uint x,y;
	for(x=xms;x>0;x--)
	 for(y=110;y>0;y--);
}
/*****初始化DS18B20*****/
void Init_DS18B20(void)
{
  unsigned char x=0;
  DQ = 1;         //DQ复位
  Delay_DS18B20(8);    //稍做延时
  DQ = 0;         //单片机将DQ拉低
  Delay_DS18B20(80);   //精确延时,大于480us
  DQ = 1;         //拉高总线
  Delay_DS18B20(14);
  x = DQ;           //稍做延时后,如果x=0则初始化成功,x=1则初始化失败
  Delay_DS18B20(20);
}
/*****读一个字节*****/
unsigned char ReadOneChar(void)
{
  unsigned char i=0;
  unsigned char dat = 0;
  for (i=8;i>0;i--)
  {
    DQ = 0;     // 给脉冲信号
    dat>>=1;
    DQ = 1;     // 给脉冲信号
    if(DQ)
    dat|=0x80;
    Delay_DS18B20(4);
  }
  return(dat);
}
/*****写一个字节*****/
void WriteOneChar(unsigned char dat)
{
  unsigned char i=0;
  for (i=8; i>0; i--)
  {
    DQ = 0;
    DQ = dat&0x01;
    Delay_DS18B20(5);
    DQ = 1;
    dat>>=1;
  }
}
/*****读取温度*****/
unsigned int ReadTemperature(void)
{
  unsigned char a=0;
  unsigned char b=0;
  unsigned int t=0;
  float tt=0;
  Init_DS18B20();
  WriteOneChar(0xCC);  //跳过读序号列号的操作
  WriteOneChar(0x44);  //启动温度转换
  Init_DS18B20();
  WriteOneChar(0xCC);  //跳过读序号列号的操作
  WriteOneChar(0xBE);  //读取温度寄存器
  a=ReadOneChar();     //读低8位
  b=ReadOneChar();    //读高8位
  t=b;
  t<<=8;
  t=t|a;
  tt=t*0.0625;
 // t= tt*10+0.5;     //放大10倍输出并四舍五入
  t= tt*10+5.5; 
  return(t);
}

/*****读取温度*****/
void check_wendu(void)
{
	c=ReadTemperature()-5;  			//获取温度值并减去DS18B20的温漂误差
	if(c>1200)
	c=1200;
}

/********液晶写入指令函数与写入数据函数,以后可调用**************/

void write_1602com(uchar com)//****液晶写入指令函数****
{
	RS=0;//数据/指令选择置为指令
//	rw=0; //读写选择置为写
	LCD1602=com;//送入数据
	delay(1);
	EN=1;//拉高使能端,为制造有效的下降沿做准备
	delay(1);
	EN=0;//en由高变低,产生下降沿,液晶执行命令
}


void write_1602dat(uchar dat)//***液晶写入数据函数****
{
	RS=1;//数据/指令选择置为数据
//	rw=0; //读写选择置为写
	LCD1602=dat;//送入数据
	delay(1);
	EN=1; //en置高电平,为制造下降沿做准备
	delay(1);
	EN=0; //en由高变低,产生下降沿,液晶执行命令
}


void lcd_init()//***液晶初始化函数****
{
	uchar a;
	write_1602com(0x38);//设置液晶工作模式,意思:16*2行显示,5*7点阵,8位数据
	write_1602com(0x0c);//开显示不显示光标
	write_1602com(0x06);//整屏不移动,光标自动右移
	write_1602com(0x01);//清显示

	write_1602com(0x80);//日历显示固定符号从第一行第1个位置之后开始显示
	for(a=0;a<16;a++)
	{
		write_1602dat(tab1[a]);//向液晶屏写日历显示的固定符号部分
		delay(3);
	}
	write_1602com(0x80+0x40);//时间显示固定符号写入位置,从第2个位置后开始显示
	for(a=0;a<16;a++)
	{
		write_1602dat(tab2[a]);//写显示时间固定符号,两个冒号
		delay(3);
	}

}

void display()
{
	if(Mode==0)
	{
		write_1602com(0x80+8);
		write_1602dat(c/1000+0x30);
		write_1602dat((c%1000)/100+0x30);
		write_1602dat(((c%1000)%100)/10+0x30);
		write_1602com(0x80+12);
		write_1602dat(((c%1000)%100)%10+0x30);
		write_1602com(0x80+13);
		write_1602dat(0xdf);
		write_1602com(0x80+0x40+3);
		write_1602dat(TH/10+0x30);
		write_1602dat(TH%10+0x30);
		write_1602dat(0xdf);
		write_1602com(0x80+0x40+12);
		write_1602dat(TL/10+0x30);
		write_1602dat(TL%10+0x30);
		write_1602dat(0xdf);		
	}								  
}
//=====================================================================================

/*****初始化定时器0*****/
void InitTimer(void)
{
	TMOD=0x1;
	TH0=0x3c;
	TL0=0xb0;     //50ms(晶振12M)
	EA=1;      //全局中断开关
	TR0=1;
	ET0=1;      //开启定时器0
}

void KEY()
{
			//功能键
	if(SET==0)
	{
		BUZZ=0;
		delay(10);
		if(SET==0)
		{
			Mode++;
			if(Mode==3)
			Mode=0;
			BUZZ=1;
		}
		while(SET==0)
		{
			if(Mode==0)
				{
				//	write_1602com(0x80+0x40+6);
					write_1602com(0x0c);
				}	
			else if(Mode==1)
				{
					write_1602com(0x80+0x40+4);
					write_1602com(0x0f);
				}	
			else
				{
					write_1602com(0x80+0x40+13);
					write_1602com(0x0f);
				}							
		}
	}
	//增加
	if(ADD==0&&Mode==1)
	{
		BUZZ=0;
		delay(10);
		if(ADD==0)	
		{
			TH++;
			if(TH>=99)	
			TH=99;
			write_1602com(0x80+0x40+3);
			write_1602dat(TH/10+0x30);
			write_1602dat(TH%10+0x30);
			write_1602com(0x80+0x40+4);	
			BUZZ=1;
		}
		while(ADD==0);
		
	}
	//减少
	if(DEC==0&&Mode==1)
	{
		BUZZ=0;
		delay(10);
		if(DEC==0)
		{
			TH--;
			if(TH==TL)	
			TH=TL+1;
			write_1602com(0x80+0x40+3);
			write_1602dat(TH/10+0x30);
			write_1602dat(TH%10+0x30);
			write_1602com(0x80+0x40+4);	
			BUZZ=1;
		}
		while(DEC==0);
	}
	if(ADD==0&&Mode==2)
	{
		BUZZ=0;
		delay(10);
		if(ADD==0)	
		{
			TL++;
			if(TL==TH)	
			TL=TH-1;
			write_1602com(0x80+0x40+12);
			write_1602dat(TL/10+0x30);
			write_1602dat(TL%10+0x30);
			write_1602com(0x80+0x40+13);	
			BUZZ=1;
		}
		while(ADD==0);
		
	}
	//减少
	if(DEC==0&&Mode==2)
	{
		BUZZ=0;
		delay(10);
		if(DEC==0)
		{
			TL--;
			if(TL<=0)	
			TL=0;
			write_1602com(0x80+0x40+12);
			write_1602dat(TL/10+0x30);
			write_1602dat(TL%10+0x30);
			write_1602com(0x80+0x40+13);	
			BUZZ=1;
		}
		while(DEC==0);		
	}
}

/*****报警子程序*****/
void Alarm()
{
	if(x>=10){beep_st=~beep_st;x=0;}
	if(Mode==0)
	{
		if((c/10)>=TH)
		{
		  fengshan=0;
			jiare=1;
			ALAM=0;
			ALAM1=1;
			if(beep_st==1)
			BUZZ=0;
			else
			BUZZ=1;
		}
		else if((c/10)<TL)
		{
			fengshan=1;
			jiare=0;
			ALAM1=0;
			ALAM=1;
			if(beep_st==1)
			BUZZ=0;
			else
			BUZZ=1;
		}
		else
		{
			BUZZ=1;
			ALAM=1;
			ALAM1=1;
			fengshan=1;	
			jiare=1;			
		}
	}
	else
	{
		BUZZ=1;
		ALAM=1;
		ALAM1=1;
		fengshan=1;
		jiare=1;
	}
}

/*****主函数*****/
void main(void)
{
	uint z;
	delay(1);
	lcd_init();
	delay(1);
	InitTimer();    //初始化定时器
	
	for(z=0;z<100;z++)
	{
		check_wendu();
		delay(1);        
	} 
	while(1)
	{
		display();
		KEY();
		Alarm(); 
		check_wendu();
	}
}

/*****定时器0中断服务程序*****/
void timer0(void) interrupt 1
{
 TH0=0x3c;
 TL0=0xb0;
 x++;
}

仿真与程序源文件百度网盘链接:https://pan.baidu.com/s/1KKbiyLIr71JHmF1Sj-sBpw 
提取码:eumg 

  • 19
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
### 回答1: 基于51单片机的电子时钟项目需要通过Keil编写程序,并使用Proteus进行仿真电路搭建。该项目主要包括以下几个步骤: 1. 硬件搭建:按照电子时钟的设计需求,连接51单片机和相关的电子元件,如晶体振荡器、数码管、按键等。通过Proteus软件,可以将这些元件连接起来,搭建出完整的电路。 2. Keil程序编写:使用Keil软件,编写51单片机的C语言程序。该程序需要实现时钟的功能,包括时、分、秒的显示和计时、调整时间、闹钟功能等。通过编程,可以控制数码管的显示,以及对按键进行响应。 3. Proteus仿真:将编写好的程序通过Proteus软件连接至搭建好的电路。进行仿真测试时,可以通过模拟时钟的不同状态,调试和验证编写的程序的正确性和稳定性。仿真过程中,可以检查数码管的显示情况,以及程序对按键输入的响应。 4. 优化和调试:根据仿真过程中的结果,对程序进行优化和调试。可能需要根据具体的需求,修改程序中的一些逻辑或代码,确保电子时钟的功能正常运行,并符合设计要求。 总的来说,基于51单片机的电子时钟项目需要通过Keil编写程序,并结合Proteus进行仿真电路搭建和测试。通过这样的开发流程,可以实现一个功能完善、稳定可靠的电子时钟。 ### 回答2: 基于51单片机的电子时钟keil程序和protues仿真电路组成了一个完整的设计方案。 首先,keil程序是用于开发51单片机的集成开发环境,它提供了编译、调试和仿真等功能,能够帮助程序员快速开发出51单片机的应用程序。在电子时钟的设计中,我们可以使用keil来编写单片机的程序代码,实现时钟的各种功能。 其次,protues是一款电子设计自动化软件,它提供了电子电路仿真和PCB布局设计等功能,能够帮助我们快速验证电路的正确性。在电子时钟的设计中,我们可以使用protues来建立电子时钟的仿真电路,验证单片机代码的正确性和稳定性。 基于51单片机的电子时钟设计,我们可以使用keil来编写单片机的程序代码,实现时钟的各种功能,包括时间显示、报时功能、闹钟功能等。通过keil的编译、调试和仿真功能,我们可以验证代码的正确性和稳定性。 在电路设计方面,我们可以使用protues来建立电子时钟的仿真电路,通过仿真可以验证电路的正确性和稳定性,包括时钟电路、数码管驱动电路、按钮输入电路等。通过protues的电路仿真功能,我们可以检查电路设计的错误和漏洞,提前解决电路问题。 基于51单片机电子时钟keil程序和protues仿真电路的组合,我们可以全面验证电子时钟的功能和性能,确保设计的准确性和可靠性,为最终实现一个完整的电子时钟设计提供了有力的支持。 ### 回答3: 基于51单片机的电子时钟keil程序和protues仿真电路可以实现以下功能: 1. 显示当前时间:我们可以使用数码管或LCD显示模块来显示当前的时、分和秒。通过编写相应的程序,我们可以从单片机的时钟源获取当前时间,并将其转换为可以在数码管或LCD上显示的格式。 2. 时间调整功能:可以通过按钮或旋钮等输入设备来调整电子时钟的时间。当用户按下或旋转输入设备时,我们可以响应用户的操作,并对时钟的时间进行相应的调整。 3. 闹钟功能:我们可以设置闹钟功能,让电子时钟在特定的时间点发出警报声。通过在程序中设置闹钟时间和警报声的播放方式,我们可以实现这一功能。 4. 温湿度监测:如果我们希望电子时钟能够同时监测室内的温度和湿度,我们可以连接温湿度传感器,并在程序中读取传感器的数据。然后,我们可以将这些数据显示在数码管或LCD上。 程序开发过程中,我们可以使用keil来编写51单片机的程序,通过keil提供的调试工具来测试和调试程序的正确性。同时,我们可以在protues设计仿真电子时钟的电路,包括单片机、显示模块、输入设备和传感器等所有的硬件组件。这样,我们可以在protues中验证电路的功能和效果,并进行性能优化和调试。当电路和程序都满足我们的要求后,我们就可以将程序烧录到实际的硬件上,并使用它作为一台完整的电子时钟了。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

薄情书生

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值