最近设计了一个项目基于单片机的鸡舍智能灯光控制系统设计,与大家分享一下:
一、基本介绍
项目名:鸡舍智能灯光控制(实物+仿真)项目编号:mcuclub-dz-007
单片机:STC89C52
功能:
1、通过光敏电阻检测光照值,当光照值低于设置最大值,进行补光(只在允许开灯时间),在最大值和最小值之间,进行PWM自动调节亮度,当低于最小值,则以最亮的开灯
2、通过DS1302时钟模块获取时间,可通过按键设置开关灯时间
3、整点保存一次光照数据,数据保存在AT24C02中(数据格式*年*月*日*时数据)
4、通过按键可以修正时间、设置开关灯时间、光照最小值、查看记录、调节不同不颜色灯进行补光
5、通过LCD1602显示时间、光照
二、51实物图
单片机型号:STC89C52
板子为绿色PCB板,两层板,厚度1.2,上下覆铜接地。元器件基本上为插针式,个别降压芯片会使用贴片式。
供电接口:TYPE-C
三、51仿真图
仿真软件版本:proteus8.9
电路连线方式:网络标号连线方式
注意:部分实物元器件仿真中没有,仿真中会用其他工作原理相似的元件代替,这样可能导致实物程序和仿真程序不一样
四、原理图
软件版本:AD2013
电路连线方式:网络标号连线方式
注意:原理图只是画出了模块的引脚图,而并不是模块的内部结构原理图
五、PCB图
由原理图导出,封装很大一部分都是作者自己绘制,不提供封装库,只提供连接好的源文件。中间有一个项目编号,隐藏在单片机底座下,插入单片机后不会看到。
两层板,上下覆铜接地。
六、系统框图
本设计以STC89C52单片机作为电路系统的核心控制器,加上其他的模块一起组成鸡舍智能灯光控制的整个系统,其中包含中控部分、输入部分和输出部分。中控部分采用了STC89C52单片机,其主要作用是获取输入部分数据,经过内部处理,控制输出部分。输入由五部分组成,第一部分是光照检测模块和A/D模数转换模块构成,用来检测当前鸡舍周围的光照强度;第二部分是存储模块,通过该模块可以存储用户设置的光照强度,设备断电后可以保存数据使得设备上电后鸡舍可以按照之前用户设置的进行控制;第三部分是时钟模块,通过该模块获取准确的时间;第四部分通过五个独立按键切换界面、设置光照强度、及其当前时间的调整、手动控制鸡舍不同灯的开关;第五部分是供电电路,给整个系统进行供电。输出由四部分组成,第一部分是LCD1602显示模块, 通过该模块可以显示当前当前的时间及其当前鸡舍周围的光照强度及其用户设置的光照最小值当信息;第二部分是继电器控制红灯;第三部分是继电器控制黄灯;第四部分是继电器控制白灯,检测到鸡舍的光照强度低于设置的最小值,自动进行鸡舍灯光的打开,用户也可以手动控制不同的鸡舍的灯光的打开。
七、软件设计流程
系统的主流程图如图4-2所示,在主程序设计流程中:考虑到调用按键扫描函数获取按键键值、对键值进行相应的处理操作(切换界面、调整定时时间、调整光照卡高度阈值),设计出五个按键函数,对系统功能进行实现。具体操作方式为:首先对各个模块进行初始化,随后进入while主循环,在主循环中,首先进入第一个函数按键函数,该函数主要分为两部分,第一部分为调用按键扫描函数获取按键键值,第二部分通过键值进行相应的处理操作,比如切换界面、调整定时时间、调整光照强度阈值、查询等;紧接着进入第二个函数监测函数,该函数主要通过调用光照检测模块和时钟模块电路获取当前的时间;紧接着进入第三个函数显示函数,该函数通过不同的界面标志位显示不同的界面,包括主界面显示当前的时间和光照值,其它界面显示设置的时间和光照值、设置定时开始时间和结束时间;紧接着进入第四个函数处理函数,测量界面,开始时间>结束时间,光照值小于设置最小值,进行补光,光照值大于等于设置最小值,停止补光。
八、部分程序展示
软件版本:keil5
逻辑程序和驱动程序分开,分布于main.c和其他.c文件
*****/
void Manage_function(void)
{
if(flag_display == 0) //测量界面
{
if(time_shi_begin*60+time_fen_begin > time_shi_end*60+time_fen_end) //开始时间>结束时间
{
if((time_shi_end*60+time_fen_end <= ds1302_buf[4]*60+ds1302_buf[5]) && (ds1302_buf[4]*60+ds1302_buf[5] < time_shi_begin*60+time_fen_begin)) //当前时间不在设置的时间内,停止补光
{
pwm_value = 0;
}
else //否则
{
if(light_value < light_min) //光照值小于设置最小值,光最亮
{
pwm_value = 10;
}
else if(light_value < light_max) //光照值小于设置最大值,根据光照强度调节补光亮度
{
pwm_value = 10 - (light_value - light_min) * (10.0 / (light_max - light_min));
}
else //光照值大于等于设置最大值,停止补光
{
pwm_value = 0;
}
}
}
else if(time_shi_begin*60+time_fen_begin < time_shi_end*60+time_fen_end) //开始时间<结束时间
{
if((time_shi_begin*60+time_fen_begin <= ds1302_buf[4]*60+ds1302_buf[5]) && (ds1302_buf[4]*60+ds1302_buf[5] < time_shi_end*60+time_fen_end)) //当前时间在设置的时间内
{
if(light_value < light_min) //光照值小于设置最小值,光最亮
{
pwm_value = 10;
}
else if(light_value < light_max) //光照值小于设置最大值,根据光照强度调节补光亮度
{
pwm_value = 10 - (light_value - light_min) * (10.0 / (light_max - light_min));
}
else //光照值大于等于设置最大值,停止补光
{
pwm_value = 0;
}
}
else //否则,停止补光
{
pwm_value = 0;
}
}
else //开始时间=结束时间,停止补光
{
pwm_value = 0;
}
}
else //设置界面,停止补光
{
pwm_value = 0;
}
if(ds1302_buf[5] == 0 && ds1302_buf[6] == 0) //整点存储时间和光照值
{
At24c02_Write_Data(data_num*8+1, ds1302_buf[1]);
Delay_function(10);
At24c02_Write_Data(data_num*8+2, ds1302_buf[2]);
Delay_function(10);
At24c02_Write_Data(data_num*8+3, ds1302_buf[3]);
Delay_function(10);
At24c02_Write_Data(data_num*8+4, ds1302_buf[4]);
Delay_function(10);
At24c02_Write_Data(data_num*8+5, ds1302_buf[5]);
Delay_function(10);
At24c02_Write_Data(data_num*8+6, ds1302_buf[6]);
Delay_function(10);
At24c02_Write_Data(data_num*8+7, ds1302_buf[7]);
Delay_function(10);
At24c02_Write_Data(data_num*8+8, light_value);
Delay_function(10);
if(data_num < 31) //最多存储31组数据
data_num++;
else
data_num = 0;
At24c02_Write_Data(0x00, data_num);
Delay_function(1000);
Ds1302_Read_Time();
}
}