使用51单片机实现SHT11温湿度传感器检测的程序和电路图

发布时间:2023-08-28  

下面是原理图:

下面是与MCU连接的典型电路:

下面是源代码:

#include 《reg52.h》

#include 《intrins.h》

/********************************************************

宏定义

********************************************************/

#define uint unsigned int

#define uchar unsigned char

#define noACK 0

#define ACK 1

#define STATUS_REG_W 0x06

#define STATUS_REG_R 0x07

#define MEASURE_TEMP 0x03

#define MEASURE_HUMI 0x05

#define RESET 0x1e

enum {TEMP,HUMI};

typedef union //定义共用同类型

{

unsigned int i;

float f;

} value;

/********************************************************

位定义

********************************************************/

sbit lcdrs=P2^0;

sbit lcdrw=P2^1;

sbit lcden=P2^2;

sbit SCK = P1^0;

sbit DATA = P1^1;

/********************************************************

********************************************************/

uchar table2[]=“SHT11 温湿度检测”;

uchar table3[]=“温度为: ℃”;

uchar table4[]=“湿度为:”;

uchar table5[]=“。”;

uchar wendu[6];

uchar shidu[6];

/********************************************************

1ms延时函数

********************************************************/

void delay(int z)

{

int x,y;

for(x=z;x》0;x--)

for(y=125;y》0;y--);

}

/********************************************************

50us延时函数

********************************************************/

void delay_50us(uint t)

{

uint j;

for(;t》0;t--)

for(j=19;j》0;j--);

}

/********************************************************

50ms延时函数

********************************************************/

void delay_50ms(uint t)

{

uint j;

for(;t》0;t--)

for(j=6245;j》0;j--);

}

/********************************************************

液晶写指令

********************************************************/

void write_12864com(uchar com)

{

lcdrs=0;

lcdrw=0;

delay_50us(1);

P0=com;

lcden=1;

delay_50us(10);

lcden=0;

delay_50us(2);

}

/********************************************************

12864液晶写数据

********************************************************/

void write_dat(uchar dat)

{

lcdrs=1;

lcdrw=0;

delay_50us(1);

P0=dat;

lcden=1;

delay_50us(10);

lcden=0;

delay_50us(2);

}

/********************************************************

12864液晶初始化

********************************************************/

void init12864lcd(void)

{

delay_50ms(2);

write_12864com(0x30);

delay_50us(4);

write_12864com(0x30);

delay_50us(4);

write_12864com(0x0f);

delay_50us(4);

write_12864com(0x01);

delay_50us(240);

write_12864com(0x06);

delay_50us(10);

write_12864com(0x0c);

delay_50us(10);

}

/********************************************************

12864液晶显示函数

********************************************************/

void display1(void)

{

uchar i;

write_12864com(0x80);

for(i=0;i《18;i++)

{

write_dat(table2[i]);

delay_50us(1);

}

}

/********************************************************

12864液晶显示函数

********************************************************/

void display2(void)

{

uchar i;

write_12864com(0x90);

for(i=0;i《18;i++)

{

write_dat(table3[i]);

delay_50us(1);

}

}

/********************************************************

12864液晶显示函数

********************************************************/

void display3(void)

{

uchar i;

write_12864com(0x88);

for(i=0;i《8;i++)

{

write_dat(table4[i]);

delay_50us(1);

}

}

/********************************************************

12864液晶显示函数

********************************************************/

void displaywendu(void)

{

uchar i;

write_12864com(0x94);

for(i=0;i《3;i++)

{

write_dat(wendu[i]);

delay_50us(1);

}

for(i=0;i《1;i++)

{

write_dat(table5[i]);

delay_50us(1);

}

for(i=4;i《5;i++)

{

write_dat(wendu[i]);

delay_50us(1);

}

}

/********************************************************

12864液晶显示函数

********************************************************/

void displayshidu(void)

{

uchar i;

write_12864com(0x8C);

for(i=0;i《3;i++)

{

write_dat(shidu[i]);

delay_50us(1);

}

for(i=0;i《1;i++)

{

write_dat(table5[i]);

delay_50us(1);

}

for(i=4;i《5;i++)

{

write_dat(shidu[i]);

delay_50us(1);

}

}

/********************************************************

SHT11写字节程序

********************************************************/

char s_write_byte(unsigned char value)

{

unsigned char i,error=0;

for (i=0x80;i》0;i》》=1) //高位为1,循环右移

{

if (i&value) DATA=1; //和要发送的数相与,结果为发送的位

else DATA=0;

SCK=1;

_nop_();_nop_();_nop_(); //延时3us

SCK=0;

}

DATA=1; //释放数据线

SCK=1;

error=DATA; //检查应答信号,确认通讯正常

_nop_();_nop_();_nop_();

SCK=0;

DATA=1;

return error; //error=1 通讯错误

}

/********************************************************

SHT11读字节程序

********************************************************/

char s_read_byte(unsigned char ack)

{

unsigned char i,val=0;

DATA=1; //释放数据线

for(i=0x80;i》0;i》》=1) //高位为1,循环右移

{

SCK=1;

if(DATA) val=(val|i); //读一位数据线的值

SCK=0;

}

DATA=!ack; //如果是校验,读取完后结束通讯;

SCK=1;

_nop_();_nop_();_nop_(); //延时3us

SCK=0;

_nop_();_nop_();_nop_();

DATA=1; //释放数据线

return val;

}

/********************************************************

SHT11启动传输

********************************************************/

void s_transstart(void)

{

DATA=1; SCK=0; //准备

_nop_();

SCK=1;

_nop_();

DATA=0;

_nop_();

SCK=0;

_nop_();_nop_();_nop_();

SCK=1;

_nop_();

DATA=1;

_nop_();

SCK=0;

}

/********************************************************

SHT11连接复位

********************************************************/

void s_connectionreset(void)

{

unsigned char i;

DATA=1; SCK=0; //准备

for(i=0;i《9;i++) //DATA保持高,SCK时钟触发9次,发送启动传输,通迅即复位

{

SCK=1;

SCK=0;

}

s_transstart(); //启动传输

}

/********************************************************

SHT11温湿度检测

********************************************************/

char s_measure(unsigned char *p_value, unsigned char *p_checksum, unsigned char mode)

{

unsigned error=0;

unsigned int i;

s_transstart(); //启动传输

switch(mode) //选择发送命令

{

case TEMP : error+=s_write_byte(MEASURE_TEMP); break; //测量温度

case HUMI : error+=s_write_byte(MEASURE_HUMI); break; //测量湿度

default : break;

}

for (i=0;i《65535;i++) if(DATA==0) break; //等待测量结束

if(DATA) error+=1; // 如果长时间数据线没有拉低,说明测量错误

*(p_value) =s_read_byte(ACK); //读第一个字节,高字节 (MSB)

*(p_value+1)=s_read_byte(ACK); //读第二个字节,低字节 (LSB)

*p_checksum =s_read_byte(noACK); //read CRC校验码

return error; // error=1 通讯错误

}

/********************************************************

SHT11温湿度值标度变换及温度补偿

********************************************************/

void calc_sth10(float *p_humidity ,float *p_temperature)

{

const float C1=-4.0; // 12位湿度精度 修正公式

const float C2=+0.0405; // 12位湿度精度 修正公式

const float C3=-0.0000028; // 12位湿度精度 修正公式

const float T1=+0.01; // 14位温度精度 5V条件 修正公式

const float T2=+0.00008; // 14位温度精度 5V条件 修正公式

float rh=*p_humidity; // rh: 12位 湿度

float t=*p_temperature; // t: 14位 温度

float rh_lin; // rh_lin: 湿度 linear值

float rh_true; // rh_true: 湿度 ture值

float t_C; // t_C : 温度 ℃

t_C=t*0.01 - 40; //补偿温度

rh_lin=C3*rh*rh + C2*rh + C1; //相对湿度非线性补偿

rh_true=(t_C-25)*(T1+T2*rh)+rh_lin; //相对湿度对于温度依赖性补偿

if(rh_true》100)rh_true=100; //湿度最大修正

if(rh_true《0.1)rh_true=0.1; //湿度最小修正

*p_temperature=t_C; //返回温度结果

*p_humidity=rh_true; //返回湿度结果

}

/********************************************************

主函数

********************************************************/

void main(void)

{

unsigned int temp,humi;

value humi_val,temp_val; //定义两个共同体,一个用于湿度,一个用于温度

unsigned char error; //用于检验是否出现错误

unsigned char checksum; //CRC

init12864lcd();

display1();

display2();

display3();

s_connectionreset(); //启动连接复位

while(1)

{

error=0; //初始化error=0,即没有错误

error+=s_measure((unsigned char*)&temp_val.i,&checksum,TEMP); //温度测量

error+=s_measure((unsigned char*)&humi_val.i,&checksum,HUMI); //湿度测量

if(error!=0) s_connectionreset(); ////如果发生错误,系统复位

else

{

humi_val.f=(float)humi_val.i; //转换为

temp_val.f=(float)temp_val.i; //转换为浮点数

calc_sth10(&humi_val.f,&temp_val.f); //修正相对湿度及温度

temp=temp_val.f*10;

humi=humi_val.f*10;

wendu[0]=temp/1000+‘0’; //温度百位

wendu[1]=temp%1000/100+‘0’; //温度十位

wendu[2]=temp%100/10+‘0’; //温度个位

wendu[3]=0x2E; //小数点

wendu[4]=temp%10+‘0’; //温度小数点后第一位

displaywendu();

shidu[0]=humi/1000+‘0’; //湿度百位

shidu[1]=humi%1000/100+‘0’; //湿度十位

shidu[2]=humi%100/10+‘0’; //湿度个位

shidu[3]=0x2E; //小数点

shidu[4]=humi%10+‘0’; //湿度小数点后第一位

displayshidu();

}

delay(800); //等待足够长的时间,以现行下一次转换

}

}


文章来源于:电子工程世界    原文链接
本站所有转载文章系出于传递更多信息之目的,且明确注明来源,不希望被转载的媒体或个人可与我们联系,我们将立即进行删除处理。

相关文章

    STM32F103ZET6读取DHT11温湿度传感器的代码: #include "dht11.h" #define DHT11_GPIO_PORT   GPIOA #define DHT11_GPIO_PIN......
    设置生成独立的初始化文件: 代码生成设置 生成代码 点击GENERATE CODE即可生成MDK-V5工程: 生成代码 3. 在MDK中编写、编译、下载用户代码 3.1. Printf重定向 在本实验中,温湿度传感器......
    STM32+DHT11 读取温湿度数据显示;一、环境介绍 MCU: STM32F103C8T6 温湿度模块: DHT11 开发软件: Keil5 二、DHT11 介绍 DHT11 数字温湿度传感器是一款含有已校准数字信号输出的温湿度复合传感器......
    申矽凌推出新一代模拟输出温湿度传感器芯片—CHT8336; 温湿度作为环境参数的重要指标,在智能化、信息化的时代对生产、生活、科研等各个领域都产生了深远的影响。温湿度传感器芯片的广泛应用不仅能够帮助用户实现温湿度......
    申矽凌推出新一代模拟输出温湿度传感器芯片—CHT8336; 温湿度作为环境参数的重要指标,在智能化、信息化的时代对生产、生活、科研等各个领域都产生了深远的影响。温湿度传感器芯片的广泛应用不仅能够帮助用户实现温湿度......
    申矽凌推出新一代模拟输出温湿度传感器芯片—CHT8336; 【导读】申矽凌推出新一代模拟输出温湿度传感器芯片-CHT8336,实现温湿度环境监测的精准化及低能耗的优化管理,为您......
    数字温湿度计设计;实验任务 任务:基于核心板 和 底板 完成计设计并观察调试结果 要求:驱动底板上的温湿度传感器SHT-20测量空气中的温度和湿度,将温湿度信息显示在8位扫描式数码管上。 解析......
    CM1241通讯模块对温湿度传感器进行通讯读取数据,使用Modbus_RTU通讯方式。温湿度传感器说明见文章尾部。 干货★★★★★资料......
    提供了设置选项,允许用户设置报警阈值参数。 四、代码设计 【1】DHT11采集温湿度 DHT11是一种数字温湿度传感器,能够通过单总线接口输出当前环境下的温度和相对湿度。它由测量模块及处理电路组成,具有......
    基于STM32的植物浇水系统开发;1.总体功能实现 本文设计的一款基于STM32的植物浇水系统,主要由STM32单片机、土壤湿度传感器温湿度传感器、Wi-Fi模块、水泵系统、继电器以及OLED液晶......

我们与500+贴片厂合作,完美满足客户的定制需求。为品牌提供定制化的推广方案、专属产品特色页,多渠道推广,SEM/SEO精准营销以及与公众号的联合推广...详细>>

利用葫芦芯平台的卓越技术服务和新产品推广能力,原厂代理能轻松打入消费物联网(IOT)、信息与通信(ICT)、汽车及新能源汽车、工业自动化及工业物联网、装备及功率电子...详细>>

充分利用其强大的电子元器件采购流量,创新性地为这些物料提供了一个全新的窗口。我们的高效数字营销技术,不仅可以助你轻松识别与连接到需求方,更能够极大地提高“闲置物料”的处理能力,通过葫芦芯平台...详细>>

我们的目标很明确:构建一个全方位的半导体产业生态系统。成为一家全球领先的半导体互联网生态公司。目前,我们已成功打造了智能汽车、智能家居、大健康医疗、机器人和材料等五大生态领域。更为重要的是...详细>>

我们深知加工与定制类服务商的价值和重要性,因此,我们倾力为您提供最顶尖的营销资源。在我们的平台上,您可以直接接触到100万的研发工程师和采购工程师,以及10万的活跃客户群体...详细>>

凭借我们强大的专业流量和尖端的互联网数字营销技术,我们承诺为原厂提供免费的产品资料推广服务。无论是最新的资讯、技术动态还是创新产品,都可以通过我们的平台迅速传达给目标客户...详细>>

我们不止于将线索转化为潜在客户。葫芦芯平台致力于形成业务闭环,从引流、宣传到最终销售,全程跟进,确保每一个potential lead都得到妥善处理,从而大幅提高转化率。不仅如此...详细>>