单片机.C51基于LCD1602光强检测仪.光敏电阻.ADC0804

														光强检测仪

#include <reg52.h>							 //头文件
#include <intrins.h>						 //_nop_()指令 12MHZ 1us

#define uint unsigned int					 //全局声明	           uint  (2^32)-1
#define uchar unsigned char					 //                    uchar (2^8)-1

sbit RGB_RED = P3^6;						 //定义RBG引脚	共阴
sbit RGB_GREEN = P3^5; 
sbit RGB_BLUE = P3^4;                         
sbit BUZZER = P3^7;							 //定义蜂鸣器  @500Hz~4.5KHz脉冲频率驱动
sbit LCD_RS = P2^5;                          //定义LDC数据命令选择端
sbit LCD_RW = P2^6;                          //定义LCD读写选择端
sbit LCD_EN = P2^7;                          //定义LCD使能端
sbit ADC_WR = P3^0;							 //定义ADC写信号输入端
sbit ADC_RD = P3^1;							 //定义ADC读信号输入端
sbit KEY_SC = P3^2;							 //外部中断0位按键位

uchar text1[] = "   UESTC&ZSC   ";
uchar text2[] = "(Lux) ";					 
uchar text3[] = "High Vau:";				 
uchar text4[] = "Low  Vau:";				 //LCD显示文本
uchar ASCII[]="0123456789";					 //ASCII
uchar Disbuf[]={0,0,0};	                     //暂存空数组 百十个位
uchar Cmax[]={0,0,0};  						 //存放最大值最小值 中断返回后重置 重新记录
uchar Cmin[]={0,0,0};
uchar CYCLE,PWM_ON;				             //T0定义周期 该数字X基准定时时间 如果是10 则周期是10 x 0.1ms,定义高电平时间 
uchar T1RH,T1RL;                             //T1 重载值的高低字节
uint max=0;min=600;							 //赋相对值 确保有比较
uint candela=0;                              //光强值
uint scale=0;                                //T1占空比控制变量

/*****************************延时函数*****************************/
 void Delay_ms(uint z)                       //毫秒ms级延时函数
 {
  uint i,j;
  for(i=z;i>0;i--)
    for(j=110;j>0;j--);
 }

void delay_us(unsigned int cnt) 			 //微秒us级延时函数
 { 
  while(--cnt); 
 } 
/******************************************************************/
 
 /***************************LCD1602驱动***************************/
 uchar Lcd1602_ReadBusy()                    //判断lCD是否处于忙的状态,即读忙
 {
  uchar temp;
  LCD_RS=0;
  LCD_RW=1;
  _nop_();
  P0=0xff;                                   //读某IO口数据前,先将该口置为1            
  _nop_();
  LCD_EN=1;
  _nop_();
  temp=P0;                                   //读取此时lcd1602的状态字
  _nop_();
  LCD_EN=0;
  return (temp&0x80);                        //最高位为1表示禁止读写,为0表示允许读写,即temp&0x80得1表示忙,得0表示不忙
 }

 void Lcd1602_WriteCom(uchar com)            //写LCD命令
 {
  while(Lcd1602_ReadBusy());                 //判忙
  LCD_RS=0;                                  //命令
  LCD_RW=0;                                  //写
  _nop_();
  P0=com;                                    //准备发送命令
  _nop_();
  LCD_EN=1;                                  //由时序图知,使能端为高电平时才允许数据交换
  _nop_();
  _nop_();
  LCD_EN=0;                                  //由时序图知,使能端在完成数据交换后要拉低
  _nop_();
  _nop_();   
 }

 void Lcd1602_WriteData(uchar dat)           //写LCD数据
 {
  while(Lcd1602_ReadBusy());                 //判忙
  LCD_RS=1;                                  //数据
  LCD_RW=0;                                  //写
  _nop_();
  P0=dat;									 //发送的是ASCII码 变量数值+0x30  /  或转码
  _nop_();                      
  LCD_EN=1;                   
  _nop_();
  _nop_();
  LCD_EN=0;
  _nop_();
  _nop_();   
 }

 void Lcd1602_init()                         //初始化LCD函数
 {
  Lcd1602_WriteCom(0x38);                    //显示模式设置
  Lcd1602_WriteCom(0x0c);                    //显示开
  Lcd1602_WriteCom(0x01);                    //显示清屏
  Lcd1602_WriteCom(0x06);                    //显示光标
 }
/******************************************************************/

/****************************ADC0804驱动***************************/
void Adc0804()					             //adc的初始化函数
 { 							
    ADC_WR=0;                                //采样
    _nop_();
    ADC_WR=1;                                //跳变高电平,触发一次ADC转换
 }

 uchar Read0804()		                     //读取ADC模数转换值
 {
  uchar r;
  P1=0xff; 				                     //P0口复位       
  _nop_();					                 //延时    
  ADC_RD=0;				                     //rd拉低
  _nop_();					                 //延时
  r=P1; 					                 //读取P0口数据
  _nop_();					                 //延时
  ADC_RD=1;				                     //rd拉高
  return(r);				                 //返回读到的数据
 }
 /******************************************************************/
 
 /****************************自定义函数****************************/ 
 void Transform(uchar dat)	                 //单片机将接收到adc0804的数据进行处理转换
 { 
  candela=dat*2.353;                         //将adc数据倍增得到一个介于0-600之间的数值
  Disbuf[0]=candela/100;		             // 百位
  Disbuf[1]=candela%100/10;                  // 十位
  Disbuf[2]=candela%10;                      // 个位     (/除取整,%除取余)
 }

 void Max_Min()								 //获得光强最大值 最小值
 {
  uchar i;
  if(candela>max)							 //光强>最大值 保存
     {
	 max=candela;
	 for(i=0;i<3;i++)
	 Cmax[i]=Disbuf[i];
	 }
  if(candela<min)						     //光强<最小值 保存
     {
	 min=candela;
	 for(i=0;i<3;i++)
	 Cmin[i]=Disbuf[i];   
 	 }
 }

 void Display_Lcd(uchar state)						 //LCD显示
 { 
  uchar i;
  switch(state)
   { 
    case 0:											 //检测显示
                 
            Lcd1602_WriteCom(0x80);					 //第一行第0位
            for(i=0;i<13;i++)
               Lcd1602_WriteData(text1[i]);			 //写入文本1
            Lcd1602_WriteCom(0x80|0x40+0x0a);		 //第二行 a(10)位
            for(i=0;i<6;i++)							 
    		   Lcd1602_WriteData(text2[i]);			 //写入文本2
  			Lcd1602_WriteCom(0x80|0x40+0x02);
 			for(i=0;i<3;i++)							 
    		   Lcd1602_WriteData(ASCII[Disbuf[i]]);
			break;
	 case 1:    									 //外部中断0后LCD显示
	        Lcd1602_WriteCom(0x80);					 //第一行第0位
            for(i=0;i<9;i++)
               Lcd1602_WriteData(text3[i]);			 //写入文本1
            Lcd1602_WriteCom(0x80|0x40);			 //第二行第0位
            for(i=0;i<9;i++)							 
    		   Lcd1602_WriteData(text4[i]);			 //写入文本2
  			Lcd1602_WriteCom(0x80+0x0a);			 
 			for(i=0;i<3;i++)							 
    		   Lcd1602_WriteData(ASCII[Cmax[i]]);	 //光强最大值
            Lcd1602_WriteCom(0x80|0x40+0x0a);
 		    for(i=0;i<3;i++)							 
    		   Lcd1602_WriteData(ASCII[Cmin[i]]);	 //光强最小值			
			break;
   }
 }

   void Buzz(uchar frequ)							 //可调频率(波形)无源蜂鸣器
 {
   static uint reload;                               //计算所需的定时器重载值	静态全局变量
   TMOD = 0x01;                                      //配置 T0 工作在模式 1
   reload = 65536-(12000000/12)/(frequ*2);           //由给定频率计算定时器重载值
   T1RH = (uchar) (reload >> 8);                     //16 位重载值分解为高低两个字节
   T1RL = (uchar) (reload);
   TH1 = 0xFF;                                       //设定一个接近溢出的初值,以使定时器马上投入工作
   TL1 = 0xFE;
   TR1 = 1;                                          //启动 T1 计时
   ET1 = 1;                                          //开启 T1 中断
 }
  
 void Display_RGB_GREEN() 							 //绿色呼吸灯
{ 
 bit Flag; 
 TMOD = 0x01;                                        //定时器设置 0.1ms in 12M crystal 
 TH0=(65536-100)/256; 
 TL0=(65536-100)%256;                                //定时0.1mS 
 EA=1;
 ET0=1;                                              //打开T0中断 
 TR0=1; 											 //开启计时
 CYCLE = 10;                                         // 时间可以调整 这个是10调整 8位PWM就是256步 
 if(!Flag) 				                            
  { 
   delay_us(3000);                                   //延时时间,从一个亮度到下一个亮度的间隔时间,速度快就能看到连续效果 
   PWM_ON++;                                         //这个使用较长延时,以便能看清楚变化过程 
   if(PWM_ON == CYCLE) 
    {                                                //可添加其他操作 到最亮时候控制对应操作 
      Flag=1; 
    } 
  } 
 if(Flag)                                            //亮度递减 相反过程 
  { 
   delay_us(3000);                                   //延迟时间为4000*0.4=1600us
   PWM_ON--; 
   if(PWM_ON == 0) 
    { 
      Flag=0; 
    }
  } 
}
 /******************************************************************/
 
 /***************************INT0 T0 T1中断*************************/
 void INT0_Max_Min() interrupt 0                      //外部中断0 显示最大值最小值
 {
  Delay_ms(300);							    
  Lcd1602_init();							          //重置LCD内容 否则进入后屏幕数据重叠乱码
  Display_Lcd(1);								      //显示内容
  while(KEY_SC);						              //中断按键按下 才能退出中断
  Lcd1602_init();						              //重置LCD内容 否则返回后屏幕数据重叠乱码
  max=0;min=600;							          //重置比较值
  Delay_ms(300);
 }

 void T0_RGB_GREEN() interrupt 1  					  //定时器T0 RBG绿呼吸灯
{ 
 static unsigned char count; 
 TH0=(65536-100)/256; 
 TL0=(65536-100)%256;                                 //重载 
 if (count==PWM_ON) 
  { 
     RGB_GREEN = 1;  								  //亮
  } 
  count++; 
 if(count == CYCLE) 
  { 
    count=0; 
    if(PWM_ON!=0)                                     //如果开启时间是0 保持原来状态 
    {
      RGB_GREEN = 0; 								  //灭
    }
  } 
}
 
 void T1_Buzz() interrupt 3				              //定时器T1 蜂鸣器发声
 {
  TH1 = T1RH;                                         //重新加载重载值
  TL1 = T1RL;
  BUZZER = ~BUZZER;                                   //反转蜂鸣器控制电平
 }
/******************************************************************/

/***************************INT0 T0 T1中断*************************/
 void main()								          //主程序
 {
  IT0=0;									          //设置中断0为低电平触发	 下降沿触发出现一直进入中断的BUG
  EX0=1;									          //INT0中断开
  EA=1;								    	          //总中断开

  Lcd1602_init();						              //lcd初始化
  
  while(1)
	{	 	
     Adc0804();               	                      //adc初始化 采样 模数转换
     Transform(Read0804());	                          //加载DB口 读取电压数字值 数据处理
     Max_Min();                                       //提取最大光强和最小光强 用于外部中断显示
	 Display_Lcd(0);                                  //lcd显示
	 			         	 
	 if(candela<=400)
	 {
	 RGB_BLUE=0;RGB_RED=0;							  //LED重置
	 ET1 = 0; TR1 = 0; 								  //关闭定时器T1(关闭蜂鸣器)
	 Display_RGB_GREEN();							  //调用绿呼吸函数
	 Delay_ms(100);
	 }

	 else if(400<candela&&candela<500) 	              //不能用400<candela<500的写法 会导致candela>=500进不去
	 {
	 RGB_RED=0;RGB_GREEN=0;RGB_BLUE=1;				  //LED重置
	 ET0=0; TR0=0;									  //关闭定时器T0(关闭绿呼吸灯)
	 ET1 = 0;TR1 = 0;                                 //关闭定时器T1(关闭蜂鸣器)	 
	 Delay_ms(200);
	 }

	 else if(candela>=500)
	 {
	 RGB_RED=~RGB_RED;RGB_GREEN=0;RGB_BLUE=0;		  //红电平反转 闪烁 LED 重置
	 ET0=0; TR0=0;                          		  //关闭定时器T0(关闭绿呼吸灯)
	 Buzz(500);									      //蜂鸣器响@1KHZ
	 Delay_ms(200);									  //适当延时调节光强变化敏感度 过短飘屏 过长迟钝
	 }	 	
	}
 }

猜你喜欢

转载自blog.csdn.net/qq_38916259/article/details/82799679