51单片机实战 4 -- sht30传感器数据读取

本次工程为一个乌龙,布置的任务为读取RS485温湿度变送器的数据,经查阅资料得知,内部核心芯片为sht30,故想到解决方案与实际出入很多。然后笔者求知若渴,不断上网找寻资料,发现很多人都做出成品,但不开源,有的需要支付RMB,有的还需要一键三连,于是,笔者决定自己编写代码。(代码少,而且简单)

 一.硬件

 笔者使用的是普中51单片机集成开发盒,sht30模块,LCD1602模块。

在此提一嘴,虽然使用的为普中集成开发盒,但可移植性还是在的,也就是说换个任意开发板,都可以套用。

二.程序代码 

 此次工程属实简单,不多逼逼,直接见代码。

#include <reg52.h>
#include <intrins.h>

#define LCD1602_DB P0   // 数据总线
sbit SCL = P2^1;        // I2C时钟线
sbit SDA = P2^0;        // I2C数据线
sbit RS = P2^6;         // LCD1602 RS引脚
sbit RW = P2^5;         // LCD1602 RW引脚
sbit EN = P2^7;         // LCD1602 EN引脚

void Delay1ms(unsigned int count) {
    unsigned int i, j;
    for(i = 0; i < count; i++)
        for(j = 0; j < 110; j++);
}

void I2C_Start() {
    SDA = 1;
    SCL = 1;
    _nop_();
    _nop_();
    SDA = 0;
    _nop_();
    _nop_();
    SCL = 0;
}

void I2C_Stop() {
    SDA = 0;
    SCL = 1;
    _nop_();
    _nop_();
    SDA = 1;
}

void I2C_SendByte(unsigned char dat) {
    unsigned char i;
    for(i = 0; i < 8; i++) {
        SDA = dat & 0x80;
        SCL = 1;
        _nop_();
        _nop_();
        SCL = 0;
        dat <<= 1;
    }
}

unsigned char I2C_ReadByte() {
    unsigned char i, dat;
    SDA = 1;
    for(i = 0; i < 8; i++) {
        SCL = 1;
        _nop_();
        _nop_();
        dat = (dat << 1) | SDA;
        SCL = 0;
    }
    return dat;
}

void LCD1602_WriteCmd(unsigned char cmd) {
    RS = 0;
    RW = 0;
    LCD1602_DB = cmd;
    EN = 1;
    Delay1ms(5);
    EN = 0;
}

void LCD1602_WriteData(unsigned char dat) {
    RS = 1;
    RW = 0;
    LCD1602_DB = dat;
    EN = 1;
    Delay1ms(5);
    EN = 0;
}

void LCD1602_Init() {
    LCD1602_WriteCmd(0x38);
    Delay1ms(5);
    LCD1602_WriteCmd(0x0c);
    Delay1ms(5);
    LCD1602_WriteCmd(0x06);
    Delay1ms(5);
    LCD1602_WriteCmd(0x01);
    Delay1ms(5);
}

void LCD1602_DisplayString(unsigned char x, unsigned char y, unsigned char *str) {
    unsigned char addr;
    if(y == 0) {
        addr = 0x80 + x;
    } else {
        addr = 0xc0 + x;
    }
    LCD1602_WriteCmd(addr);
    while(*str != '\0') {
        LCD1602_WriteData(*str);
        str++;
    }
}

void main() {
    unsigned char temperature[5], humidity[5];
    
    I2C_Start();
    I2C_SendByte(0x44);
    I2C_SendByte(0x2f);
    I2C_Stop();
    Delay1ms(50);
    
    LCD1602_Init();
    
    while(1) {
        I2C_Start();
        I2C_SendByte(0x45);
        I2C_SendByte(0x00);
        I2C_SendByte(0x00);
        I2C_Stop();
        Delay1ms(260);
        
        I2C_Start();
        I2C_SendByte(0x45);
        humidity[0] = I2C_ReadByte();
        I2C_SendByte(0xff);
        humidity[1] = I2C_ReadByte();
        I2C_SendByte(0x2f);
        temperature[0] = I2C_ReadByte();
        I2C_SendByte(0xff);
        temperature[1] = I2C_ReadByte();
        I2C_Stop();
        
        LCD1602_DisplayString(0, 0, "Temp:");
        LCD1602_DisplayString(6, 0, temperature);
        LCD1602_DisplayString(0, 1, "Humidity:");
        LCD1602_DisplayString(9, 1, humidity);
        
        Delay1ms(1000);  // 延时1秒
    }
}

IIC通讯协议要搞清楚,时序控制是关键,对于LCD并没有调用库,直接从地址写,可能是由于笔者个人习惯,从底层写,才更容易移植,不会出现差异化,接线就不附上了,因为笔者没有做实物,但基本功能已经实现。 

sht30的基于c51单片机驱动程序:#include #include #include "I2C.h" #include "SHT30.h" #define uint unsigned int #define uchar unsigned char void display(); unsigned char code tableduan[]= { 0x3f,0x06,0x5b,0x4f, 0x66,0x6d,0x7d,0x07, 0x7f,0x6f,0x77,0x7c, 0x39,0x5e,0x79,0x71 }; uchar data DIS_ROME[6]= {0,0,0,0,0,0}; //显示缓存区(4) uchar DISP=0;//缓存区指针 uchar SCANF=0xDF;//扫描指针 sbit LED1=P1^0; sbit LED2=P1^1; sbit LED3=P1^2; sbit LED4=P1^3; sbit VOC_A=P3^5; sbit VOC_B=P3^6; sbit dula=P2^6; //IO口定义 sbit wela=P2^7; sbit key=P3^4; sbit beep_dr=P2^3; uint pm1 = 0; uint pm2 = 0; uint pm10 = 0; uchar vr=0; uint intrcnt=0; bit F_1HZ; uint voice_time_cnt; uchar Uart_Buf; uchar Rec_Addr=0; uchar mode=0; uchar Rec_Uart=0; uchar Recive_Buf[30]= {0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0}; #define key P34 #define const_key_time1 50 unsigned char ucKeySec=0; //被触发的按键编号 unsigned int uiKeyTimeCnt1=0; //按键去抖动延时计数器 unsigned char ucKeyLock1=0; //按键触发后自锁的变量标志 unsigned char displaycnt=0; void keyscan() { if(key==1)//IO是高电平,说明按键没有被按下,这时要及时清零一些标志位 { ucKeyLock1=0; //按键自锁标志清零 uiKeyTimeCnt1=0;//按键去抖动延时计数器清零,此行非常巧妙,是我实战中摸索出来的。 } else if(ucKeyLock1==0)//有按键按下,且是第一次被按下 { uiKeyTimeCnt1++; //累加定时中断次数 if(uiKeyTimeCnt1>const_key_time1) { uiKeyTimeCnt1=0; ucKeyLock1=1; //自锁按键置位,避免一直触发 ucKeySec=1; //触发1号键 } } } void keyservice() { if(ucKeySec) { displaycnt=!displaycnt; } ucKeySec=0; } void UartInit(void) //9600bps@12.000MHz { TMOD=0x01; //设置定时器0为工作方式1 TH0=0xf8; //重装初始值(65535-500)=65035=0xfe0b TL0=0x2f; SCON=0x50; TMOD=0X21; IP =0x10; //把串口中断设置为最高优先级, EA=1; ES=1; ET0=1; TR0=1; } void T0_time(void) interrupt 1 //定时中断 { TF0=0; //清除中断标志 TR0=0; //关中断 keyscan(); keyservice(); display(); if(++intrcnt==1000) { intrcnt=0; } TH0=0xf8; TL0=0x2f; TR0=1; //开中断 } void display() //LED扫描 { if(displaycnt==1) { DIS_ROME[0]=0; DIS_ROME[1]=Hum_num[4]; DIS_ROME[2]=Hum_num[3]; DIS_ROME[3]=Hum_num[2]; DIS_ROME[4]=Hum_num[1]; DIS_ROME[5]=Hum_num[0]; } else { DIS_ROME[0]=0; DIS_ROME[1]=TEMP_num[4]; DIS_ROME[2]=TEMP_num[3]; DIS_ROME[3]=TEMP_num[2]; DIS_ROME[4]=TEMP_num[1]; DIS_ROME[5]=TEMP_num[0]; } wela=1; P0=SCANF; wela=0; dula=1; P0=tableduan[DIS_ROME[DISP]];//数据端口送数据 dula=0; DISP++;//缓存指针加1 SCANF=_cror_(SCANF,1);//扫描切换 if(DISP==7)//缓存指针到尾 { DISP=0;//计数归零 SCANF=0xDF;//扫完四个数码管,扫描复位 } // delay(5); } void main(void) //主函数 { UartInit(); I2C_inita(); while(1) { Getdat_SHT30(); SHT30_DATEChange(); } }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

下一站是上岸

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值