[创新制造展示] 【21ic DIY秀】+ 智能调光台灯

[复制链接]
5825|12
 楼主| hbzjt2011 发表于 2019-2-27 12:56 | 显示全部楼层 |阅读模式
本帖最后由 hbzjt2011 于 2019-2-27 12:56 编辑

智能调光台灯

1、作品介绍:
        闲来无事,重拾51,制作了一个基于51单片机的智能调光台灯,主要实现了以下功能:两种工作模式,一种自动调节的,还有一种手动按键调节。自动模式时,白天自动灭灯,晚上自动亮灯,灯光可以根据周围光强自动调节,晚上有人自动亮灯,没人延时一段时间自动熄灭。手动模式时,可以通过按键手动调节灯光亮度。作品简陋,纯当抛砖引玉了。

2、作品方案及主要元件介绍:
         作品方案:作品的系统框图如下所示,主要采用了STC89C52单片机最小系统用作系统的控制器,采用热释电红外人体感应传感器作为系统的人体检测,判断周围是否有人;采用光敏电阻用于光线和强度检测,ADC0832数模转换芯片采集光敏电阻的电压值进行转换以后传给单片机进行处理;单片机通过光强信号使用PWM进行LED灯的亮度调节,同时使用三个单独按键负责模式切换、手动模式亮度增加和手动模式亮度减小的功能。

         元件清单及主要元器件介绍:
STC89C52单片机:STC89C52RC单片机是宏晶科技推出的新一代高速/低功耗/超强抗干扰的单片机,指令代码完全兼容传统8051单片机,12时钟/机器周期和6时钟/机器周期可以任意选择。增强型8051单片机,6时钟/机器周期和12时钟/机器周期可以任意选择,指令代码完全兼容传统8051。工作电压:5.5V~3.3V(5V单片机)/3.8V~2.0V(3V单片机)。工作频率范围:0~40MHz,相当于普通8051的0~80MHz,实际工作频率可达48MH。 用户应用程序空间为8K字节,片上集成512字节RAM。
热释电红外感应模块:HC-SR501 是基于红外线技术的自动控制模块, 采用德国原装进口LHI778 探头设计,灵敏度高,可靠性强,超低电压工作模式,广泛应用于各类自动感应电器设备,尤其是干电池供电的自动控制产品。主要特性:1、全自动感应:人进入其感应范围则输出高电平,人离开感应范围则自动延时关闭高电平,输出低电平。2、光敏控制(可选择,出厂时未设)可设置光敏控制,白天或光线强时不感应。3、温度补偿(可选择,出厂时未设):在夏天当环境温度升高至30~32℃,探测距离稍变短,温度补偿可作一定的性能补偿。4、两种触发方式。5、具有感应封锁时间。
光敏电阻:型号:GL5528  
最大电压(V-dc):150  
最大功耗(mW):100  
环境温度(°C):-30--- +70  
光谱峰值(nm):540  
亮电阻(10Lux)(KΩ):10-20  
暗电阻(MΩ):1  
100λ10:0.6  
响应时间(ms):
                上升:20                    
                下降:30
ADC0832芯片介绍:ADC0832 为 8 位分辨率 A/D 转换芯片,其最高分辨可达 256 级,可以适应一般的模拟量转换要求。其内部电源输入与参考电压的复用,使得芯片的模拟电压输入在 0~5V 之间。芯片转换时间仅为 32μS,据有双数据输出可作为数据校验,以减少数据误差,转换速度快且稳定性能强。独立的芯片使能输入,使多器件挂接和处理器控制变的更加方便。通过 DI 数据输入端,可以轻易的实现通道功能的选择。
3、原理图设计:
         系统的硬件原理图如下:

4、程序设计:
        软件程序流程图如下:
程序源码:
  1. //宏定义
  2. #define uint unsigned int
  3. #define uchar unsigned char
  4. //头函数
  5. #include <reg52.h>               
  6. uchar pdata tt[51];                   //定义空数组用于AD0832取平均值

  7. uchar scale,rsd_sec;                //定义占空比比例,热释电计时秒变量
  8. uchar sec;
  9. bit bdata flag_auto,flag_rsd;          //位定义自动切换,热释电动作标志

  10. uint lum;                                        //ad0832读出值
  11. uchar m;

  12. //管脚声明
  13. sbit LED = P2^5;        //灯光控制输出
  14. sbit rsd = P1^5;   //热释电
  15. sbit qiehuan=P3^7;
  16. sbit add=P3^6;
  17. sbit dec=P3^5;

  18. sbit CS=P1^3;                //CS定义脚,连接ADC0832CS脚
  19. sbit SCL=P1^0;                //SCL定义脚,连接ADC0832SCL脚
  20. sbit DO=P1^1;                //DO定义脚,连接ADC0832DO脚

  21. /*****************延时函数:大约1ms************************/
  22. void delay(uchar i)
  23. {
  24.   uchar j,k;
  25.   for(j=i;j>0;j--)
  26.     for(k=125;k>0;k--);
  27. }
  28. /***********读数模转换数据********************************************************/        
  29. //请先了解ADC0832模数转换的串行协议,再来读本函数,主要是对应时序图来理解,本函数是模拟0832的串行协议进行的

  30. unsigned char ad0832read(bit SGL,bit ODD)
  31. {
  32.         unsigned char i=0,value=0,value1=0;               
  33.                 SCL=0;
  34.                 DO=1;
  35.                 CS=0;                //开始
  36.                 SCL=1;                //第一个上升沿        
  37.                 SCL=0;
  38.                 DO=SGL;
  39.                 SCL=1;          //第二个上升沿
  40.                 SCL=0;
  41.                 DO=ODD;
  42.                 SCL=1;            //第三个上升沿
  43.                 SCL=0;            //第三个下降沿
  44.                 DO=1;
  45.                 for(i=0;i<8;i++)
  46.                 {
  47.                         SCL=1;
  48.                         SCL=0; //开始从第四个下降沿接收数据
  49.                         value<<=1;
  50.                         if(DO)
  51.                                 value++;                                                
  52.                 }
  53.                 for(i=0;i<8;i++)
  54.                 {                        //接收校验数据
  55.                         value1>>=1;
  56.                         if(DO)
  57.                                 value1+=0x80;
  58.                         SCL=1;
  59.                         SCL=0;
  60.                 }
  61.                 CS=1;
  62.                 SCL=1;        
  63.                 if(value==value1)                                //与校验数据比较,正确就返回数据,否则返回0        
  64.                         return value;
  65.         return 0;
  66. }

  67. /*****************按键函数*****************/
  68. void work()
  69. {
  70.         uint lum_mean,lum_all;
  71.         uchar b,c;

  72.         if(qiehuan==0)                                  //自动切换按键按下
  73.         {
  74.                 delay(10);                                  //去抖
  75.                 if(qiehuan==0)                          //再次判断按键按下
  76.                 {
  77.                         flag_auto=!flag_auto; //自动模式标志位取反
  78.                         if(flag_auto==1)          //当切换到手动模式时  首先将LED发光比例设置在50%
  79.                         scale=20;
  80.                 }
  81.                 while(!qiehuan); //按键释放  松开按键后关闭蜂鸣器
  82.         }
  83.         if(flag_auto==1)
  84.         {
  85.                 if(add==0)                                 //加键按下
  86.                 {
  87.                         delay(10);
  88.                         if(add==0)
  89.                         {
  90.                                 scale++;                  //灯光比例++
  91.                                 if(scale>=41)
  92.                                 scale=41;
  93.                         }
  94.                 }
  95.                 if(dec==0)                                  //减键按下时
  96.                 {
  97.                         delay(10);
  98.                         if(dec==0)
  99.                         {
  100.                                 scale--;                   //灯光比例--
  101.                                 if(scale<=1)
  102.                                 scale=1;
  103.                         }
  104.                 }
  105.         }
  106.         else
  107.         {
  108.                 if(flag_rsd==1)                                        //有人在范围内时
  109.                 {         
  110.                         for(b=0;b<49;b++)                         //将空数组tt【】内数值整体左移一位
  111.                         {
  112.                                 tt[b]=tt[b+1];                        //将后一数值放到前一位置
  113.                         }        
  114.                         tt[49] = ad0832read(1,0);  //将读出的ad数值放入tt【49】
  115.                         for(c=0;c<50;c++)                        //将tt【】内数值相加
  116.                         {
  117.                                 lum_all=lum_all+tt[c];
  118.                         }
  119.                         lum_mean=lum_all/50;                //将总数/50取出平均值
  120.                         lum_all=0;                                        //将总数清零
  121.                         if(lum_mean<=30) scale=1;                           //判断取出平均值大小  小于30  发光强度0%
  122.                         else if(lum_mean>=150) scale=41;           //大于150  发光强度100%
  123.                         else scale=((lum_mean-30)/3)+1;                   //其他值时将其计算得到发光强度 (计算目的是为了得到一个1-41之间的数值 控制灯光变化)        
  124.                 }
  125.                 else
  126.                 {
  127.                         scale=1;                                         //没有人在范围内时 将灯光亮度调至0%
  128.                 }        
  129.         }
  130. }
  131. /*********定时器初始化函数**********/
  132. void init()
  133. {
  134.         TMOD=0x11;           //工作方式
  135.         TH1=0x3c;
  136.         TL1=0xb0;           //T1赋初值50ms
  137.         TH0=0xff;
  138.         TL0=0xe7;                //T0赋初值25us         
  139.         ET0=1;
  140.         ET1=1;                   //打开中断允许开关
  141.         EA=1;                   //中断总开关
  142.         TR0=1;
  143.         TR1=1;                   //定时器定时开关
  144. }

  145. /****************主函数**********************/
  146. void main()
  147. {
  148.         init();                  //调用初始化函数

  149.         while(1)                   //循环
  150.         {
  151.                 work();                 //调用函数
  152.         }
  153. }

  154. /******************定时器T0服务函数:脉冲发生函数*******************/
  155. void time0() interrupt 1
  156. {
  157.         uchar n;
  158.         TH0=0xff;
  159.         TL0=0xe7;                 //重新赋初值
  160.         n++;                         //每25us  n++
  161.         if(n<scale)                 //n<设置比例时,打开灯
  162.         {
  163.                 LED=0;
  164.         }
  165.         else if(n>=scale)//n大于等于设置比例时 关闭灯
  166.         {
  167.                 LED=1;
  168.         }
  169.         if(n>=40)                 //n==40  :25us*40=1ms   1kHZ
  170.         {
  171.                 n=0;                 //n=0
  172.         }
  173. }  
  174. /********************定时器T0服务函数:计时和闪烁控制********************/
  175. void time1() interrupt 3
  176. {
  177.         TH1=0x3c;
  178.         TL1=0xb0;                 //重新赋初值
  179.         m++;                         //50ms  m++
  180.         
  181.         if(m==20)                                                  //到达1s时
  182.         {
  183.                 m=0;                                                  //m=0
  184.                 if(rsd==0)                                          //热释电无信号时
  185.                 rsd_sec++;                                          //热释电计时秒++
  186.                 if(rsd_sec<=20&&rsd==1)                  //热释电计时秒小于等于20 并且 热释电有信号时
  187.                 {
  188.                         rsd_sec=0;                                  //将热释电秒清零
  189.                         flag_rsd=1;                                  //标志位置1 控制AD0832采集数值 调节灯光亮度
  190.                 }
  191.                 else if(rsd_sec>20&&rsd==0)          //热释电计时秒大于20 并且 热释电无信号时
  192.                 {
  193.                         flag_rsd=0;                                  //标志位置0 停止ad0832转换 关闭灯光
  194.                         rsd_sec=0;                                  //热释电计时秒清零
  195.                 }
  196.         }
  197. }

5、制作过程:
焊接过程比较简单,元器件购买,万用板焊接:

6、作品展示:
焊接完成后的效果如下:
视频效果展示:


本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?注册

×
duhemayi 发表于 2019-2-27 17:14 | 显示全部楼层
shunde117 发表于 2019-2-28 09:13 | 显示全部楼层
喜欢!!!!
second_chan 发表于 2019-2-28 09:41 | 显示全部楼层
感谢楼主分享。。。
 楼主| hbzjt2011 发表于 2019-2-28 09:49 | 显示全部楼层

谢谢支持
 楼主| hbzjt2011 发表于 2019-2-28 09:49 | 显示全部楼层
second_chan 发表于 2019-2-28 09:41
感谢楼主分享。。。

谢谢支持
百川入海 发表于 2019-2-28 11:28 | 显示全部楼层
楼主做的很不错,谢谢分享。
jarye 发表于 2019-3-6 14:08 | 显示全部楼层
菜鸟才像你这样做,红外热释感应器只感应移动带热源的目标,你坐在灯下读书或写作业等不动的时候分分钟就熄灭,体验不好。单片机都有自带ADC的,还去加片IC,而且光敏二极管是不精确的,DIY一下没问题,做产品想要做的很精致光敏二极管搞不定,需要用光传感器。
 楼主| hbzjt2011 发表于 2019-3-7 14:19 | 显示全部楼层
jarye 发表于 2019-3-6 14:08
菜鸟才像你这样做,红外热释感应器只感应移动带热源的目标,你坐在灯下读书或写作业等不动的时候分分钟就熄 ...

呵呵,大神批评的是,既然都说了是DIY,何必在意那么多,再说单片机都带ADC,恐怕没这个说法吧
一路向北lm 发表于 2019-3-7 15:06 | 显示全部楼层
jarye 发表于 2019-3-6 14:08
菜鸟才像你这样做,红外热释感应器只感应移动带热源的目标,你坐在灯下读书或写作业等不动的时候分分钟就熄 ...

需要鼓励,别那么强势,明白就好
ningling_21 发表于 2019-3-9 14:05 | 显示全部楼层
hbzjt2011 发表于 2019-3-7 14:19
呵呵,大神批评的是,既然都说了是DIY,何必在意那么多,再说单片机都带ADC,恐怕没这个说法吧 ...

楼主做的不错,有遥控就更好了
liushiming82 发表于 2019-3-13 16:46 | 显示全部楼层
动手能力强,顶起
character 发表于 2020-9-22 08:20 | 显示全部楼层
很好啊
您需要登录后才可以回帖 登录 | 注册

本版积分规则

个人签名:欢迎参与LabVIEW版块的讨论学习! 点我一键即达

323

主题

2908

帖子

45

粉丝
快速回复 在线客服 返回列表 返回顶部