C51小项目 - 温控智能风扇
做一个简单的环境温湿度感应与显示器
电路连接分析
Pin | 名称 | 注释 |
---|---|---|
1 | VDD | 供电3-5V |
2 | DATA | 串行数据(单总线) |
3 | NC | 空脚(悬空) |
4 | GND | 接地 |
需要注意的是,该实验选择P3^6引脚位DATA数据引脚,其他引脚可以修改
数据分析
· 数据采集及时序分析
DHT11采用单总线通信,单总线即只有一根数据线,系统中的数据交换、控制均由单总线完成。
· 传送数据位定义
DATA 管脚用于DHT11与单片机之间的通讯和同步,采用单总线数据格式,一次传送40 位数据,高位先出。
· 数据格式8bit 湿度整数数据 + 8bit 湿度小数数据 + 8bit 温度整数数据 + 8bit 温度小数数据 + 8bit 校验位
注:其中湿度小数部分为0
· 校验位数据定义
8bit 湿度整数数据 + 8bit 湿度小数数据 + 8bit 温度整数数据 + 8bit 温度小数数据 = 8bit 校验位,如果以上等式成立,则本次传感器采集的数据有效,否则无效。
· 数据位的定义
0011 0101 | 0000 0000 | 0001 1000 | 0000 0100 | 0101 0001 |
湿度高8位 | 湿度低8位 | 温度高8位 | 温度低8位 | 校验位 |
计算:
0011 0101 + 0000 0000 + 0001 1000 + 0000 0100 = 0101 0001,接收数据正确。
湿度:
0011 0101(整数)=35H=53%RH 0000 0000(小数)=00H=0.0%RH =>53%RH + 0.0%RH = 53.0%RH
温度:
0001 1000(整数)=18H=24℃ 0000 0100(小数)=04H=0.4℃ =>24℃ + 0.4℃ = 24.4℃
时序分析
用户主机(MCU)发送一次开始信号后,DHT11 从低功耗模式转换到高速模式,待主机开始信号结束后,DHT11 发送响应信号,送出 40bit 的数据,幵触发一次信采集,依次轮回
单片机连接DHT11的DATA引脚的I/O口输出低电平,且低电平保持时间不能小于 18ms,然后等待 DHT11 作出应答信号。
DHT11 的 DATA 引脚检测到外部信号有低电平时, 等待外部信号低电平结束, 延迟后 DHT11 的 DATA引脚处于输出状态,输出 80 微秒的低电平作为应答信号,紧接着输出 80 微秒的高电平通知外设准备接收数据。
位数据“0”的格式为: 50 微秒的低电平和 26-28 微秒的高电平,位数据“1”的格式为: 50 微秒的低电平加 70微秒的高电平。
代码分析
· main.c
#include<reg51.h>
#include"lcd.h"
#include<intrins.h>
#include<stdio.h>
sbit Temp_data=P3^6;
//unsigned char data_byte;
//unsigned char rec_dat_lcd=[4];
unsigned int rec_dat[4];
unsigned char rec_dat_lcd0[6];
unsigned char rec_dat_lcd1[6];
unsigned char rec_dat_lcd2[6];
unsigned char rec_dat_lcd3[6];
void DHT11_delay_us(unsigned char n);
void DHT11_delay_ms(unsigned int z);
void InitUART(void);
void DHT11_start();
void DHT11_receive();
unsigned char DHT11_rec_byte();
//主函数
void main() {
//unsigned char i,j;
InitUART();
P1=0xf0;
InitLcd1602();
LcdShowStr(0,0,"Humi:");
LcdShowStr(0,1,"Temp:");
EA = 1; //开总中断
while(1) {
DHT11_delay_ms(150);
DHT11_receive();
sprintf(rec_dat_lcd0,"%d",rec_dat[0]);
sprintf(rec_dat_lcd1,"%d",rec_dat[1]);
sprintf(rec_dat_lcd2,"%d",rec_dat[2]);
sprintf(rec_dat_lcd3,"%d",rec_dat[3]);
DHT11_delay_ms(100);
//湿度
LcdShowStr(6,0,rec_dat_lcd0);
LcdShowStr(8,0,".");
LcdShowStr(9,0,rec_dat_lcd1);
LcdShowStr(10,0," %");
//温度
LcdShowStr(6,1,rec_dat_lcd2);
LcdShowStr(8,1,".");
LcdShowStr(9,1,rec_dat_lcd3);
LcdShowStr(10,1," C");
//下面通过串口助手打印温度
printf("Humi:%d.%d \n",rec_dat[0],rec_dat[1]);
printf("Temp:%d.%d °C\n",rec_dat[2],rec_dat[3]);
}
}
//DHT11起始信号
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;
}
}
//延时us --2*n+5us
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--);
}
//使用定时器1作为串口波特率发生器
void InitUART(void) {
SCON=0x40; //串口通信工作方式1
REN=1; //允许接收
TMOD=0x20; //定时器1的工作方式2
TH1=0xF3,TL1=0xF3;
TI=1; //这里一定要注意
TR1=1;
}
· Lcd.c
#include"lcd.h"
//void Read_Busy() { //忙检测函数,判断bit7是0,允许执行;1禁止
// unsigned char sta;
// LCD1602_DB = 0xff;
// LCD1602_RS = 0;
// LCD1602_RW = 1;
// do {
// LCD1602_EN = 1;
// sta = LCD1602_DB;
// LCD1602_EN = 0; //使能,用完就拉低,释放总线
// }while(sta & 0x80);
//}
//写命令
void Lcd1602_Write_Cmd(unsigned char cmd) {
//Read_Busy();
LCD1602_RS = 0;
LCD1602_RW = 0;
LCD1602_DB = cmd;
LCD_Delay10ms(1);
LCD1602_EN = 1;
LCD_Delay10ms(1);
LCD1602_EN = 0;
}
//写数据
void Lcd1602_Write_Data(unsigned char dat) {
//Read_Busy();
LCD1602_RS = 1;
LCD1602_RW = 0;
LCD1602_DB = dat;
LCD_Delay10ms(1);
LCD1602_EN = 1;
LCD_Delay10ms(1);
LCD1602_EN = 0;
}
//指定位置开始显示数据!
//坐标显示
void LcdSetCursor(unsigned char x,unsigned char y) {
unsigned char addr;
if(y == 0)
addr = 0x00 + x;//第一行开始,x表示一行的第x个
else
addr = 0x40 + x;//第二行开始,x表示一行的第x个
Lcd1602_Write_Cmd(addr|0x80);
}
//显示字符串
void LcdShowStr(unsigned char x,unsigned char y,unsigned char *str) {
LcdSetCursor(x,y); //当前字符的坐标
while(*str != '\0') {
Lcd1602_Write_Data(*str++);
}
}
//1602初始化
void InitLcd1602() {
Lcd1602_Write_Cmd(0x38); //打开,5*8,8位数据
Lcd1602_Write_Cmd(0x0c);
Lcd1602_Write_Cmd(0x06);
Lcd1602_Write_Cmd(0x01); //清屏
}
//误差 0us
void LCD_Delay10ms(unsigned int c) {
unsigned char a,b;
for(;c>0;c--)
for(b=38;b>0;b--)
for(a=130;a>0;a--);
}
· Lcd.h
#ifndef __LCD_H_
#define __LCD_H_
/**********************************
当使用的是4位数据传输的时候定义
使用8位取消这个定义
#define LCD1602_4PINS
**********************************/
#include<reg51.h>
//---重定义关键词---//
#ifndef uchar
#define uchar unsigned char
#endif
#ifndef uint
#define uint unsigned int
#endif
/**********************************
PIN口定义
**********************************/
#define LCD1602_DB P0 //data bus 数据总线
sbit LCD1602_RS = P2^6;
sbit LCD1602_RW = P2^5;
sbit LCD1602_EN = P2^7;
/**********************************
函数声明
**********************************/
/*在51单片机12MHZ时钟下的延时函数*/
//void Lcd1602_Delay1ms(uint c); //误差 0usvo
void LCD_Delay10ms(unsigned int c);
//void Read_Busy(); //忙检测函数,判断bit7是0,允许执行;1禁止
void Lcd1602_Write_Cmd(unsigned char cmd); //写命令
void Lcd1602_Write_Data(unsigned char dat); //写数据
void LcdSetCursor(unsigned char x,unsigned char y); //坐标显示
void LcdShowStr(unsigned char x,unsigned char y,unsigned char *str); //显示字符串
void InitLcd1602(); //1602初始化
#endif
实验问题
在实验中一开始,并未能正常的在LCD1602上显示,反而出现乱码,此时分析原因为,显示的数据类型不对。在此过程中,不断通过串口打印DHT11传回来的数据进行调试最后才成功。其中,printf()串口打印又成一个问题,通过学习发现一个很好的方法
首先导入#include
后来查询得知:
12M的晶振波特率只能是2400,9600的情况下会有7.8%的误差,所以会产生乱码,而2400波特率的情况下误差是0.16%,这样就不会产生乱码了,因此TH1和TL1都设为F3
void InitUART(void) { //使用定时器1作为串口波特率发生器
SCON=0x40; //串口通信工作方式1
REN=1; //允许接收
TMOD=0x20; //定时器1的工作方式2
TH1=0xF3,TL1=0xF3;
TI=1; //这里一定要注意
TR1=1;
}
注意:因为本实验使用的晶振是12M,如果你的晶振是11.0592,将TH1和TL1设置为0xFd,波特率=9600
成功打印出温度数据后,接下来就是将温度数据在LCD1602上显示了,于是,使用sprintf()函数进行字符串拼接,进行显示,当然这只是我当时想到的方法,如果你有更好的方法也可以的!
本作品采用 知识共享署名-非商业性使用-禁止演绎 4.0 国际许可协议 (CC BY-NC-ND 4.0) 进行许可。