找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 2493|回复: 4
打印 上一主题 下一主题
收起左侧

基于51单片机的智能台灯带坐姿矫正Proteus仿真设计(包含仿真和源程序)

  [复制链接]
跳转到指定楼层
楼主
功能描述:
1、系统分为自动和手动模式,上电之后,绿色的LED灯点亮,代表当前是自动模式。
2、通过按键1,可以在手动模式(绿灯灭)和自动模式(绿灯亮)之间来回切换。
3、在手动模式下,可以通过按键2降低亮度,按键3增加亮度。
4、在自动模式下,首先要检测到有人,才进行亮度的自动控制,否则台灯熄灭。调整光敏电阻,模拟光照的变化,从而就可以看到台灯的亮度会随着环境中的光照强度的改变而改变,效果是光照越弱,台灯越亮。如果人体传感器 1 分钟检测不到有人,台灯就会自动熄灭。
5、无论是自动模式还是手动模式,都是把亮度分为 10 个等级的。
6、注意,人体感应检测功能只在自动模式下有效。
7、带坐姿矫正功能,使用超声波模块测量距离,距离过近发出告警,提醒摆正坐姿。

仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)


单片机源程序如下:
  1. #include <reg52.h>
  2. #include <intrins.h>
  3. #include "ultrasonic_wave.h"//超声波头函数

  4. #define uchar unsigned char                // 以后unsigned char就可以用uchar代替
  5. #define uint  unsigned int                // 以后unsigned int 就可以用uint 代替


  6. sbit LED     = P1^0;                                        // 模式指示灯,亮是自动模式,灭是手动模式
  7. sbit Lamp    = P1^4;                                         // 台灯控制引脚
  8. sbit Key1    = P1^1;                                        // 按键1,模式切换按键
  9. sbit Key2    = P1^2;                                         // 按键2,亮度减少按键      
  10. sbit Key3    = P1^3;                                        // 按键3,亮度增加按键
  11. sbit ADC_CS  = P2^3;                                         // ADC0832的CS引脚
  12. sbit ADC_CLK = P2^0;                                         // ADC0832的CLK引脚
  13. sbit ADC_DAT = P2^1;                                         // ADC0832的DI/DO引脚
  14. sbit Module  = P2^2;                                        // 人体红外检测模块
  15. sbit Beep    = P1^5;                            // 蜂鸣器引脚定义

  16. uchar gCount=0;                                                                // 全局计数变量
  17. uchar gIndex;                                                                        // 亮度变量,0是最暗,9是最亮,一共10档
  18. uint  gTime=0;                                                                // 计时变量,用于计时多久没检测到有人



  19. /*********************************************************/
  20. // 毫秒级的延时函数,time是要延时的毫秒数
  21. /*********************************************************/
  22. void DelayMs(uint time)
  23. {
  24.         uint i,j;
  25.         for(i=0;i<time;i++)
  26.                 for(j=0;j<112;j++);
  27. }



  28. /*********************************************************/
  29. // ADC0832的时钟脉冲
  30. /*********************************************************/
  31. void WavePlus()
  32. {
  33.         _nop_();
  34.         ADC_CLK = 1;
  35.         _nop_();
  36.         ADC_CLK = 0;
  37. }



  38. /*********************************************************/
  39. // 获取指定通道的A/D转换结果
  40. /*********************************************************/
  41. uchar Get_ADC0832()
  42. {
  43.         uchar i;
  44.         uchar dat1=0;
  45.         uchar dat2=0;
  46.         
  47.         ADC_CLK = 0;                                // 电平初始化
  48.         ADC_DAT = 1;
  49.         _nop_();
  50.         ADC_CS = 0;
  51.         WavePlus();                                        // 起始信号
  52.         ADC_DAT = 1;
  53.         WavePlus();                                        // 通道选择的第一位
  54.         ADC_DAT = 0;      
  55.         WavePlus();                                        // 通道选择的第二位
  56.         ADC_DAT = 1;
  57.         
  58.         for(i=0;i<8;i++)                // 第一次读取
  59.         {
  60.                 dat1<<=1;
  61.                 WavePlus();
  62.                 if(ADC_DAT)
  63.                         dat1=dat1|0x01;
  64.                 else
  65.                         dat1=dat1|0x00;
  66.         }
  67.         
  68.         for(i=0;i<8;i++)                // 第二次读取
  69.         {
  70.                 dat2>>= 1;
  71.                 if(ADC_DAT)
  72.                         dat2=dat2|0x80;
  73.                 else
  74.                         dat2=dat2|0x00;
  75.                 WavePlus();
  76.         }
  77.         
  78.         _nop_();                                                // 结束此次传输
  79.         ADC_DAT = 1;
  80.         ADC_CLK = 1;
  81.         ADC_CS  = 1;   

  82.         if(dat1==dat2)                        // 返回采集结果
  83.                 return dat1;
  84.         else
  85.                 return 0;
  86. }



  87. /*********************************************************/
  88. // 定时器初始化
  89. /*********************************************************/
  90. void TimerInit()
  91. {
  92.         TMOD = 0x01;                                // 使用定时器0,工作方式1         
  93.         TH0  = 252;                                        // 给定时器0的TH0装初值
  94.         TL0  = 24;                                        // 给定时器0的TL0装初值        
  95.         ET0  = 1;                                                // 定时器0中断使能
  96.         EA   = 1;                                                // 打开总中断
  97.         TR0         = 1;                                                // 启动定时器0
  98. }



  99. /*********************************************************/
  100. // 手动控制
  101. /*********************************************************/
  102. void ManualControl()
  103. {
  104.         // 亮度减少
  105.         if(Key2==0)                                        // 如果按键2被按下去
  106.         {
  107.                 if(gIndex>0)                        // 只要当前亮度不为最低才能减少亮度
  108.                 {
  109.                         gIndex--;                                // 亮度降低一档
  110.                         DelayMs(300);                // 延时0.3秒
  111.                 }
  112.         }
  113.         
  114.         // 亮度增加
  115.         if(Key3==0)                                        // 如果按键3被按下去
  116.         {
  117.                 if(gIndex<9)                        // 只要当前亮度不为最高才能增加亮度
  118.                 {
  119.                         gIndex++;                                // 亮度增加一档
  120.                         DelayMs(300);                // 延时0.3秒
  121.                 }
  122.         }
  123. }



  124. /*********************************************************/
  125. // 自动控制
  126. /*********************************************************/
  127. void AutoControl(uchar num)
  128. {
  129.         if(num<59)                                                                                                                // 最亮
  130.                 gIndex=9;
  131.         else if((num>65)&&(num<81))                                                // 第二亮
  132.                 gIndex=8;
  133.         else if((num>87)&&(num<103))                                        // 第三亮
  134.                 gIndex=7;
  135.         else if((num>109)&&(num<125))
  136.                 gIndex=6;
  137.         else if((num>131)&&(num<147))
  138.                 gIndex=5;
  139.         else if((num>153)&&(num<169))
  140.                 gIndex=4;
  141.         else if((num>175)&&(num<191))
  142.                 gIndex=3;
  143.         else if((num>197)&&(num<213))
  144.                 gIndex=2;
  145.         else if((num>219)&&(num<235))
  146.                 gIndex=1;
  147.         else if(num>241)                                                                                 // 最暗
  148.                 gIndex=0;
  149. }



  150. /*********************************************************/
  151. // 主函数
  152. /*********************************************************/
  153. void main()
  154. {
  155.         uchar ret;
  156.         
  157.         TimerInit();                                         // 定时器初始化
  158.         Init_ultrasonic_wave();          //超声波定时器初始化
  159.         LED=0;                                                // 指示灯点亮(自动模式指示灯)
  160.         ret=Get_ADC0832();                // 获取AD采集结果(环境光照强度)
  161.         AutoControl(ret);                        // 上电先进行一次自动亮度控制        
  162.         AutoControl(ret+7);
  163.         
  164.         while(1)
  165.         {

  166.             StartModule();//启动超声波
  167.                 while(!RX)          //当RX为零时等待
  168.                 TR1=1;                  //开启计数
  169.                 while(RX);          //当RX为1计数并等待
  170.                 TR1=0;                  //关闭计数
  171.                 DelayMs(20);
  172.                                 
  173.                 Conut(); //计算距离
  174.                    //距离小于30
  175.                 if(L_ < 30)
  176.                 {                                                                 
  177. //                     Beep=~Beep;
  178. //                         DelayMs(1);        
  179. //                         Beep=~Beep;
  180. //                         DelayMs(1);
  181.                          Beep=0;
  182.                          DelayMs(200);
  183.        Beep=1;
  184.        DelayMs(200);                        
  185.                 }
  186.                 else
  187.                 Beep=1;


  188.                 /* 模式切换控制 */
  189.                 if(Key1==0)                                        // 如果按键1被按下去
  190.                 {
  191.                         LED=~LED;                                        // 切换LED灯状态
  192.                         DelayMs(10);                        // 延时消除按键按下的抖动
  193.                         while(!Key1);                        // 等待按键释放
  194.                         DelayMs(10);                        // 延时消除按键松开的抖动
  195.                 }
  196.                         
  197.                 /* 亮度控制 */
  198.                 if(LED==1)                                                        // 如果LED是灭的
  199.                 {
  200.                         ManualControl();                        // 则进行手动控制
  201.                         DelayMs(200);
  202.                 }
  203.                 else                                                                                // 如果LED是亮的
  204.                 {
  205.                         if(gTime<60000)
  206.                         {
  207.                                 ret=Get_ADC0832();                // 获取AD采集结果(环境光照强度)
  208.                                 AutoControl(ret);                        // 进行自动控制        
  209.                                 DelayMs(200);
  210.                         }
  211.                 }
  212.                
  213.                 /*检测是否有人*/
  214.                 if(Module==1)
  215.                 {
  216.                         gTime=0;                                                                                // 检测到有人,则把60秒计时清零
  217.                 }
  218.                 if(gTime>60000)                                                                // 如果gTime的值超过了60000
  219.                 {
  220.                         gTime=60000;                                                                // 则把gTime的值重新赋值为60000,避免过大溢出
  221.                         gIndex=0;                                                                                // 如果1分钟检测不到有人,则把台灯熄灭
  222.                 }
  223.         }
  224. }


  225. /*********************************************************/
  226. // 定时器0服务程序,1毫秒
  227. /*********************************************************/
  228. void Timer0(void) interrupt 1
  229. {
  230.         TH0  = 252;                                                // 给定时器0的TH0装初值
  231.         TL0  = 24;                                                // 给定时器0的TL0装初值        
  232.         
  233.         gTime++;                                                        // 每1毫秒,gTime变量加1
  234.         gCount++;                                                        // 每1毫秒,gCount变量加1
  235.         
  236.         if(gCount==10)                                // 如果gCount加到10了
  237.         {
  238.                 gCount=0;                                                // 则将gCount清零,进入新一轮的计数
  239.                 if(gIndex!=0)                                // 如果说台灯不是最暗的(熄灭)
  240.                 {
  241.                         Lamp=0;                                                // 则把台灯点亮
  242.                 }
  243.         }
  244.         if(gCount==gIndex)                // 如果gCount计数到和gIndex一样了
  245.         {
  246.                 if(gIndex!=9)                                // 如果说台灯不是最亮的
  247.                 {
  248.                         Lamp=1;                                                // 则把台灯熄灭
  249.                 }
  250.         }
  251. }


  252. void time1() interrupt 3                  //T1中断用来计数器溢出,超过测距范围
  253. {
  254.         TH1=0;
  255.         TL1=0;
  256. }  
复制代码

Keil代码与Proteus8.13仿真下载:
仿真和源程序.7z (180.12 KB, 下载次数: 171)

评分

参与人数 1黑币 +50 收起 理由
admin + 50 共享资料的黑币奖励!

查看全部评分

分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏10 分享淘帖 顶1 踩
回复

使用道具 举报

沙发
ID:1065040 发表于 2023-3-28 16:07 | 只看该作者
大佬,我参考了你的设计,把实物做出来了,但是取消了红外热释电感应,大部分功能都能实现,但是光敏电阻调光却实现不了,这是为什么呢
回复

使用道具 举报

板凳
ID:1065040 发表于 2023-3-30 00:29 | 只看该作者
补充,以上大佬程序和仿真是完全没有问题的,如果想要实物做出来,我出现的状况是无论自动还是手动模式的灯是否亮起,都只能采集通电瞬间的灯光,后面只能手动调节。解决办法是我在按钮KEY1翻转那里设置了flag,把 if(LED==1) 这里的条件分成两个while(flag==0)和while(flag==1),即不以LED为标志,这样是我遇到的问题以及解决办法。有同样情况的同学可以参考一下。
回复

使用道具 举报

地板
ID:1102013 发表于 2023-12-1 13:19 | 只看该作者
为啥会宏文件嵌套过多呢?头几行
回复

使用道具 举报

5#
ID:1101012 发表于 2023-12-4 11:59 | 只看该作者
写的很好
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表