显示出来是这样的
这是主函数:
- #include <reg52.h>
- #include "LCD1602.h"
- #include "stdio.h"
- unsigned char str1[2],str2[2];
- bit flag1s = 0; //1s定时标志
- unsigned char T0RH = 0; //T0重载值的高字节
- unsigned char T0RL = 0; //T0重载值的低字节
- void ConfigTimer0(unsigned int ms);
- extern void UartDriver();
- extern void ConfigUART(unsigned int baud);
- extern void UartRxMonitor(unsigned char ms);
- extern void UartWrite(unsigned char *buf, unsigned char len);
- extern unsigned char RH_data,TH_data;
- void main ()
- {
- EA = 1; //开总中断
- ConfigTimer0(1); //配置T0定时1ms
- ConfigUART(9600); //配置波特率为9600
- InitLcd1602(); //初始化液晶
- DHT11_start();
- LcdShowStr(0,0,"Humi:");
- LcdShowStr(0,1,"Temp:");
- while (1)
- {
- UartDriver(); //调用串口驱动
- if(flag1s)
- {
- flag1s=0;
- DHT11_receive();
- str1[0]=(char)(0x30+RH_data/10); //湿度整数部分
- str1[1]=(char)(0x30+RH_data%10);
- str2[0]=(char)(0x30+TH_data/10); //温度整数部分
- str2[1]=(char)(0x30+TH_data%10);
- LCD_Write_Char(6,0,str1[0]);
- LCD_Write_Char(7,0,str1[1]);
- LcdShowStr(9,0,"%");
- LCD_Write_Char(6,1,str1[0]);
- LCD_Write_Char(7,1,str1[1]);
- LcdShowStr(9,1,"C");
- }
- }
- }
- /* 串口动作函数,根据接收到的命令帧执行响应的动作
- buf-接收到的命令帧指针,len-命令帧长度 */
- void UartAction(unsigned char *buf, unsigned char len)
- {
- // buf[len++] = '\r'; //有效命令被执行后,在原命令帧之后添加
- // buf[len++] = '\n'; //回车换行符后返回给上位机,表示已执行
- // UartWrite(buf, len);
- //UartWrite(buf, len);
- }
- /* 配置并启动T0,ms-T0定时时间 */
- void ConfigTimer0(unsigned int ms)
- {
- unsigned long tmp; //临时变量
-
- tmp = 11059200 / 12; //定时器计数频率
- tmp = (tmp * ms) / 1000; //计算所需的计数值
- tmp = 65536 - tmp; //计算定时器重载值
- tmp = tmp + 32; //补偿中断响应延时造成的误差
- T0RH = (unsigned char)(tmp>>8); //定时器重载值拆分为高低字节
- T0RL = (unsigned char)tmp;
- TMOD &= 0xF0; //清零T0的控制位
- TMOD |= 0x01; //配置T0为模式1
- TH0 = T0RH; //加载T0重载值
- TL0 = T0RL;
- ET0 = 1; //使能T0中断
- TR0 = 1; //启动T0
- }
- /* T0中断服务函数,执行串口接收监控和蜂鸣器驱动 */
- void InterruptTimer0() interrupt 1
- {
- static unsigned char tmr1s = 0;
- TH0 = T0RH; //重新加载重载值
- TL0 = T0RL;
- UartRxMonitor(1); //串口接收监控
- tmr1s++;
- if (tmr1s >= 100) //定时1s
- {
- tmr1s = 0;
- flag1s = 1;
- }
- }
- 这是DHT11程序:
- #include "DHT11.h"
- //sbit Temp_data=P2^1;
- unsigned char RH_data,TH_data;
- unsigned int rec_dat[4];
- //DHT11起始信号
- void DHT11_delay_us(unsigned char n)
- {
- while(--n);
- }
- //延时ms
- void DHT11_delay_ms(unsigned int z)
- {
- unsigned int i,j;
- for(i=z;i>0;i--)
- for(j=110;j>0;j--);
- }
- void DHT11_start()
- {
- Temp_data=1;
- DHT11_delay_us(2);
- Temp_data=0;
- DHT11_delay_ms(20);
- Temp_data=1;
- DHT11_delay_us(13);
-
- }
- //接收一个字节
- unsigned char DHT11_rec_byte()
- {
- unsigned char i,dat;
- for(i=0;i<8;i++)
- {
- while(!Temp_data);
- DHT11_delay_us(8);
- dat<<=1;
- if(Temp_data==1)
- {
- dat+=1;
- }
- while(Temp_data);
- }
- return dat;
- }
- //接收温湿度数据
- void DHT11_receive()
- {
- unsigned int R_H,R_L,T_H,T_L;
- unsigned char RH,RL,TH,TL,revise;
-
- DHT11_start();
- Temp_data=1;
- if(Temp_data==0)
- {
- while(Temp_data==0); //等待拉高
- DHT11_delay_us(40); //拉高后延时80us
-
- R_H=DHT11_rec_byte(); //接收湿度高八位
- R_L=DHT11_rec_byte(); //接收湿度低八位
- T_H=DHT11_rec_byte(); //接收温度高八位
- T_L=DHT11_rec_byte(); //接收温度低八位
- revise=DHT11_rec_byte(); //接收校正位
- DHT11_delay_us(25); //结束
- if((R_H+R_L+T_H+T_L)==revise) //校正
- {
- RH=R_H;
- RL=R_L;
- TH=T_H;
- TL=T_L;
-
- }
- /*数据处理,方便显示*/
- /* rec_dat[0]=RH;
- rec_dat[1]=RL;
- rec_dat[2]=TH;
- rec_dat[3]=TL;*/
- RH_data=RH;
- TH_data=TH;
- }
- }
- 这是Uart.c程序:
- #include <reg52.h>
- bit flagFrame = 0; //帧接收完成标志,即接收到一帧新数据
- bit flagTxd = 0; //单字节发送完成标志,用来替代TXD中断标志位
- unsigned char cntRxd = 0; //接收字节计数器
- unsigned char pdata bufRxd[64]; //接收字节缓冲区
- extern void UartAction(unsigned char *buf, unsigned char len);
- /* 串口配置函数,baud-通信波特率 */
- void ConfigUART(unsigned int baud)
- {
- SCON = 0x50; //配置串口为模式1
- TMOD &= 0x0F; //清零T1的控制位
- TMOD |= 0x20; //配置T1为模式2
- TH1 = 256 - (11059200/12/32)/baud; //计算T1重载值
- TL1 = TH1; //初值等于重载值
- ET1 = 0; //禁止T1中断
- ES = 1; //使能串口中断
- TR1 = 1; //启动T1
- }
- /* 串口数据写入,即串口发送函数,buf-待发送数据的指针,len-指定的发送长度 */
- void UartWrite(unsigned char *buf, unsigned char len)
- {
- while (len--) //循环发送所有字节
- {
- flagTxd = 0; //清零发送标志
- SBUF = *buf++; //发送一个字节数据
- while (!flagTxd); //等待该字节发送完成
- }
- }
- /* 串口数据读取函数,buf-接收指针,len-指定的读取长度,返回值-实际读到的长度 */
- unsigned char UartRead(unsigned char *buf, unsigned char len)
- {
- unsigned char i;
-
- if (len > cntRxd) //指定读取长度大于实际接收到的数据长度时,
- { //读取长度设置为实际接收到的数据长度
- len = cntRxd;
- }
- for (i=0; i<len; i++) //拷贝接收到的数据到接收指针上
- {
- *buf++ = bufRxd[i];
- }
- cntRxd = 0; //接收计数器清零
-
- return len; //返回实际读取长度
- }
- /* 串口接收监控,由空闲时间判定帧结束,需在定时中断中调用,ms-定时间隔 */
- void UartRxMonitor(unsigned char ms)
- {
- static unsigned char cntbkp = 0;
- static unsigned char idletmr = 0;
- if (cntRxd > 0) //接收计数器大于零时,监控总线空闲时间
- {
- if (cntbkp != cntRxd) //接收计数器改变,即刚接收到数据时,清零空闲计时
- {
- cntbkp = cntRxd;
- idletmr = 0;
- }
- else //接收计数器未改变,即总线空闲时,累积空闲时间
- {
- if (idletmr < 30) //空闲计时小于30ms时,持续累加
- {
- idletmr += ms;
- if (idletmr >= 30) //空闲时间达到30ms时,即判定为一帧接收完毕
- {
- flagFrame = 1; //设置帧接收完成标志
- }
- }
- }
- }
- else
- {
- cntbkp = 0;
- }
- }
- /* 串口驱动函数,监测数据帧的接收,调度功能函数,需在主循环中调用 */
- void UartDriver()
- {
- unsigned char len;
- unsigned char pdata buf[40];
- if (flagFrame) //有命令到达时,读取处理该命令
- {
- flagFrame = 0;
- len = UartRead(buf, sizeof(buf)); //将接收到的命令读取到缓冲区中
- UartAction(buf, len); //传递数据帧,调用动作执行函数
- }
- }
- /* 串口中断服务函数 */
- void InterruptUART() interrupt 4
- {
- if (RI) //接收到新字节
- {
- RI = 0; //清零接收中断标志位
- if (cntRxd < sizeof(bufRxd)) //接收缓冲区尚未用完时,
- { //保存接收字节,并递增计数器
- bufRxd[cntRxd++] = SBUF;
- }
- }
- if (TI) //字节发送完毕
- {
- TI = 0; //清零发送中断标志位
- flagTxd = 1; //设置字节发送完成标志
- }
- }
复制代码 |