stm32+oled+dht11

今天搞了一个小实验。基于stm32f103c8t6的温湿度检测,话不多说,直接进入重要部分。

1、模块介绍

1、stm32f103c8t6,不多介绍了,网上资源丰富(喜欢偷懒哈哈)。

2、dht11,不多介绍了,网上资源丰富(喜欢偷懒哈哈)。

顺带提一句:dht11模块因为是单总线传输,所以在开发时候一定要注意看时序图(切记切记,本人踩了坑)。

3、oled,我这次用的是四针的、并且支持IIC通信协议。IIC其实也没啥好说的,别的博主介绍的非常详细。SCL跟SDA进行软件模拟通信即可。

2、部分模块例程

OLED.c

/**
  * @brief  OLED显示一个字符
  * @param  Line 行位置,范围:1~4
  * @param  Column 列位置,范围:1~16
  * @param  Char 要显示的一个字符,范围:ASCII可见字符
  * @retval 无
  */
void OLED_ShowChar(uint8_t Line, uint8_t Column, char Char)
{      	
	uint8_t i;
	OLED_SetCursor((Line - 1) * 2, (Column - 1) * 8);		//设置光标位置在上半部分
	for (i = 0; i < 8; i++)
	{
		OLED_WriteData(OLED_F8x16[Char - ' '][i]);			//显示上半部分内容
	}
	OLED_SetCursor((Line - 1) * 2 + 1, (Column - 1) * 8);	//设置光标位置在下半部分
	for (i = 0; i < 8; i++)
	{
		OLED_WriteData(OLED_F8x16[Char - ' '][i + 8]);		//显示下半部分内容
	}
}

/**
  * @brief  OLED显示字符串
  * @param  Line 起始行位置,范围:1~4
  * @param  Column 起始列位置,范围:1~16
  * @param  String 要显示的字符串,范围:ASCII可见字符
  * @retval 无
  */
void OLED_ShowString(uint8_t Line, uint8_t Column, char *String)
{
	uint8_t i;
	for (i = 0; String[i] != '\0'; i++)
	{
		OLED_ShowChar(Line, Column + i, String[i]);
	}
}

/**
  * @brief  OLED次方函数
  * @retval 返回值等于X的Y次方
  */
uint32_t OLED_Pow(uint32_t X, uint32_t Y)
{
	uint32_t Result = 1;
	while (Y--)
	{
		Result *= X;
	}
	return Result;
}

/**
  * @brief  OLED显示数字(十进制,正数)
  * @param  Line 起始行位置,范围:1~4
  * @param  Column 起始列位置,范围:1~16
  * @param  Number 要显示的数字,范围:0~4294967295
  * @param  Length 要显示数字的长度,范围:1~10
  * @retval 无
  */
void OLED_ShowNum(uint8_t Line, uint8_t Column, uint32_t Number, uint8_t Length)
{
	uint8_t i;
	for (i = 0; i < Length; i++)							
	{
		OLED_ShowChar(Line, Column + i, Number / OLED_Pow(10, Length - i - 1) % 10 + '0');
	}
}

/**
  * @brief  OLED显示数字(十进制,带符号数)
  * @param  Line 起始行位置,范围:1~4
  * @param  Column 起始列位置,范围:1~16
  * @param  Number 要显示的数字,范围:-2147483648~2147483647
  * @param  Length 要显示数字的长度,范围:1~10
  * @retval 无
  */
void OLED_ShowSignedNum(uint8_t Line, uint8_t Column, int32_t Number, uint8_t Length)
{
	uint8_t i;
	uint32_t Number1;
	if (Number >= 0)
	{
		OLED_ShowChar(Line, Column, '+');
		Number1 = Number;
	}
	else
	{
		OLED_ShowChar(Line, Column, '-');
		Number1 = -Number;
	}
	for (i = 0; i < Length; i++)							
	{
		OLED_ShowChar(Line, Column + i + 1, Number1 / OLED_Pow(10, Length - i - 1) % 10 + '0');
	}
}

/**
  * @brief  OLED显示数字(十六进制,正数)
  * @param  Line 起始行位置,范围:1~4
  * @param  Column 起始列位置,范围:1~16
  * @param  Number 要显示的数字,范围:0~0xFFFFFFFF
  * @param  Length 要显示数字的长度,范围:1~8
  * @retval 无
  */
void OLED_ShowHexNum(uint8_t Line, uint8_t Column, uint32_t Number, uint8_t Length)
{
	uint8_t i, SingleNumber;
	for (i = 0; i < Length; i++)							
	{
		SingleNumber = Number / OLED_Pow(16, Length - i - 1) % 16;
		if (SingleNumber < 10)
		{
			OLED_ShowChar(Line, Column + i, SingleNumber + '0');
		}
		else
		{
			OLED_ShowChar(Line, Column + i, SingleNumber - 10 + 'A');
		}
	}
}

/**
  * @brief  OLED显示数字(二进制,正数)
  * @param  Line 起始行位置,范围:1~4
  * @param  Column 起始列位置,范围:1~16
  * @param  Number 要显示的数字,范围:0~1111 1111 1111 1111
  * @param  Length 要显示数字的长度,范围:1~16
  * @retval 无
  */
void OLED_ShowBinNum(uint8_t Line, uint8_t Column, uint32_t Number, uint8_t Length)
{
	uint8_t i;
	for (i = 0; i < Length; i++)							
	{
		OLED_ShowChar(Line, Column + i, Number / OLED_Pow(2, Length - i - 1) % 2 + '0');
	}
}

OLED.h 

#ifndef __OLED_H
#define __OLED_H

void OLED_Init(void);
void OLED_Clear(void);
void OLED_ShowChar(uint8_t Line, uint8_t Column, char Char);
void OLED_ShowString(uint8_t Line, uint8_t Column, char *String);
void OLED_ShowNum(uint8_t Line, uint8_t Column, uint32_t Number, uint8_t Length);
void OLED_ShowSignedNum(uint8_t Line, uint8_t Column, int32_t Number, uint8_t Length);
void OLED_ShowHexNum(uint8_t Line, uint8_t Column, uint32_t Number, uint8_t Length);
void OLED_ShowBinNum(uint8_t Line, uint8_t Column, uint32_t Number, uint8_t Length);

#endif

DHT11.c

void DHT11_Reset(void)
{
	DHT11_IO_OUT();//输出模式
	DHT11_DQ_OUT=0;//引脚拉低
	Delay_ms(20);//延时18ms以上
	DHT11_DQ_OUT=1;//引脚拉高
	Delay_us(30);//延时30us
}


//检测函数,return0:接收成功
//          return1:接收失败
u8 DHT11_Check(void)//检测函数
{
	u8 temp=0;//临时变量
	
	DHT11_IO_IN(); 
	while(DHT11_DQ_IN&&temp<100)
	{
		Delay_us(1);
		temp++;
	};
	if(temp>=100) return 1;
	else temp=0;//清零
	while(!DHT11_DQ_IN&&temp<100)
	{
		Delay_us(1);
		temp++;
	};
	if(temp>=100) return 1;
	return 0;//DHT11
}

u8 DHT11_Read_Bit(void)//从DHT11读一个位
{
	u8 temp=0;
	
	//DHT11_Pin_IN();//输入模式
	while(DHT11_DQ_IN&&temp<100)//等待变为低电平
	{
		Delay_us(1);
		temp++;
	}
	temp=0;
	while(!DHT11_DQ_IN&&temp<100)//等待变为高电平
	{
		Delay_us(1);
		temp++;
	}
	Delay_us(40);
	if(DHT11_DQ_IN)
		return 1;
	else 
		return 0;
}
u8 DHT11_Read_Byte(void)
{
	u8 i=0;
	u8 dat=0;
	
	for(i=0;i<8;i++)//高位到低位,数据位左移或等运算即可
	{
		dat<<=1;
		dat|=DHT11_Read_Bit();
	}
	return dat;
}

u8 DHT11_Read_Data(u8 *temp,u8 *humi)//温度,湿度
{
	u8 buf[5];
	u8 i=0;
	DHT11_Reset();//复位函数,开始读取温湿度
	
	if(DHT11_Check()==0)//为0则检查成功
	{
		for(i=0;i<5;i++)//读取40位数据
		{
			buf[i]=DHT11_Read_Byte();//读出shuju
			
		}
		if(buf[4]==(buf[0]+buf[1]+buf[2]+buf[3]))//数据校验成功
		{
			*temp = buf[2];//读取温度
			*humi = buf[0];//读取湿度		
		}
		else return 1;
	}

	return 0;
	
}

u8 DHT11_Init(void)//初始化的DHT11
{
	GPIO_InitTypeDef GPIO_InitStructure;
	
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);//开启端口时钟
	
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;//推挽输出
	GPIO_InitStructure.GPIO_Pin = DHT11_Pin;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(DHT11_PORT,&GPIO_InitStructure);
	DHT11_Reset();
	return DHT11_Check();//返回检查结果
}

DHT11.h 

#ifndef _DHT11_H
#define _DHT11_H

#include "sys.h"

#define DHT11_RCC       RCC_APB2Periph_GPIOA
#define DHT11_PORT      GPIOA
#define DHT11_Pin       GPIO_Pin_11
//IO方向设置
#define DHT11_IO_IN()  {GPIOA->CRH&=0XFFFF0FFF;GPIOA->CRH|=8<<12;}//输入
#define DHT11_IO_OUT() {GPIOA->CRH&=0XFFFF0FFF;GPIOA->CRH|=3<<12;}//推挽输出50MHZ
//IO操作函数
#define	DHT11_DQ_OUT PAout(11) //数据端口	PA11
#define	DHT11_DQ_IN  PAin(11)  //数据端口	PA11

//DHT11函数定义
u8 DHT11_Init(void);


void DHT11_Reset(void);
u8 DHT11_Check(void);
u8 DHT11_Read_Bit(void);
u8 DHT11_Read_Byte(void);
u8 DHT11_Read_Data(u8 *temp,u8 *humi);
#endif













3、 总结

       这次小实验就是个非常非常非常............非常简单的实验,可能连个课设都算不上,在家没啥事,写个博客玩玩。

     有什么不懂的可以留言或者私信,可以交流解答。


版权声明:本文为qq_46686576原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明。