找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 1443|回复: 4
收起左侧

运用光敏电阻实现自动调光的单片机程序为啥不能进入自动调光了?求大佬解答

[复制链接]
ID:912956 发表于 2021-5-3 15:36 | 显示全部楼层 |阅读模式
调试电路如下:

利用进行手动调光,但是一旦按下P3.7切换开关后,波形图直接变成直线了,无法进行自动调光,希望大佬帮我看看。

单片机程序如下:

  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. }
复制代码

回复

使用道具 举报

ID:435676 发表于 2021-5-3 20:01 来自手机 | 显示全部楼层
自己好好在看清楚一下程序把!不提倡这种写法!这样写主程序没有灵魂!何必要这么麻烦,ADC的精度也太差了!才8位的芯片自带的都可以了!U2电路不对哦!这样取永远是一个固定值!不会变的!这样程序在牛逼也是没用的!我不太明白你的电路具体情况!电路设计不对!

评分

参与人数 1黑币 +20 收起 理由
admin + 20 回帖助人的奖励!

查看全部评分

回复

使用道具 举报

ID:554500 发表于 2021-5-4 08:35 | 显示全部楼层
C:\Users\Administrator\Desktop

用ADC0831吧,单通道的

#include "ADC1031.H"



void delay1(u16 x)
{
        while(x--);
}


u8 read_0831() //读取TLC0831的数字量
{
        u8 d,i;
       
        ck_0831=0;
        cs_0831=0;
        delay1(50);
        ck_0831=1;
        ck_0831=0; //此时0831输出一个虚假的0
        for(i=0;i<8;i++)
        {
                 ck_0831=1;
                ck_0831=0;
                d<<=1;   //dddd dddd==>> dddd ddd0
                if(do_0831==1)
                {
                         d++;
                }
        }

        ck_0831=0;
        cs_0831=1;
               
        return d;        
}




#ifndef _ADC1031_
#define _ADC1031_

#include"reg51.h"

#define u8 unsigned char
#define u16 unsigned int
#define u32 unsigned long
       
sbit cs_0831=P1^5;
sbit do_0831=P1^7;
sbit ck_0831=P1^6;

extern u8 read_0831(void); //读取TLC0831的数字量;

#endif



回复

使用道具 举报

ID:912956 发表于 2021-5-13 17:37 | 显示全部楼层
18701931930 发表于 2021-5-4 08:35
用ADC0831吧,单通道的

#include "ADC1031.H"

把模数转换程序换成这个就行了吗,
回复

使用道具 举报

ID:912956 发表于 2021-5-13 17:39 | 显示全部楼层
1051798957 发表于 2021-5-3 20:01
自己好好在看清楚一下程序把!不提倡这种写法!这样写主程序没有灵魂!何必要这么麻烦,ADC的精度也太差了 ...

这个是某一个宝买的,我不会改程序,怕越改越差,这里就是ADC连线错误了吗,那应该怎么连呀
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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