51单片机上连YL69土壤湿度传感器获取的数据在LCD上显示出来

要做一个项目,被分配到做DS18B20温度传感与YL69土壤湿度传感器在51单片机上用LCD显示屏显示出来。温度传感模块很简单,网上到处都是资料,但是YL69的资料就很少了,特别还是在51单片机上实现。
其实懂了原理也还是简单。将传感器的AO口与单片机的ADDA模块通道3相连,将会实现数字量转换为模拟量,GND接地,VCC接电源(这里我们不接DO口,不会影响)。接下来就是代码部分。
此代码实现的是,YL69获取的数据在液晶显示屏上显示出来

#include<reg52.h>
#include<intrins.h>
#include<math.h>
typedef unsigned char uchar	  ;
typedef unsigned int uint	   ;
#define AT24C02_ADDR  0xa0	//AT24C02地址
#define DAC_EN 0x40
#define ADC_AutoINC 0x04
#define PCF8591_ADDR  0x90	//PCF8591地址
#define MAIN_Fosc		11059200L	//定义主时钟
sbit SDA = P2^0;
sbit SCL = P2^1;
sbit LCDEN=P3^4;
sbit LCD_RS	= P3^5;			//LCD写数据或命令控制IO
sbit LCD_RW = P3^6;			//LCD读写控制IO
sbit dula = P2^6;
sbit wela = P2^7;
uint strHEX[4];INT8UtostrHEX函数转换的字符串显示10进制
void delay_ms(uint z)
{
	uint x,y;
	for(x = z; x > 0; x--)
		for(y = 114; y > 0 ; y--);
}


void delay_5us()  
{
	_nop_();
}

/*I2C初始化*/
void I2C_init()	
{
	SDA = 1;
	_nop_();
	SCL = 1;
	_nop_();
}

/*I2C起始信号*/
void I2C_Start()  
{
	SCL = 1;
	_nop_();
	SDA = 1;
	delay_5us();
	SDA = 0;
	delay_5us();
}

/*I2C终止信号*/
void I2C_Stop()
{
	SDA = 0;
	_nop_();
	SCL = 1;
	delay_5us();
	SDA = 1;
	delay_5us();
}
/*主机发送应答*/
void Master_ACK(bit i)		
{
	SCL = 0; // 拉低时钟总线允许SDA数据总线上的数据变化
	_nop_(); // 让总线稳定
	if (i)	 //如果i = 1 那么拉低数据总线 表示主机应答
	{
		SDA = 0;
	}
	else	 
	{
		SDA = 1;	 //发送非应答
	}
	_nop_();//让总线稳定
	SCL = 1;//拉高时钟总线 让从机从SDA线上读走 主机的应答信号
	delay_5us();
	SCL = 0;//拉低时钟总线, 占用总线继续通信
	_nop_();
	SDA = 1;//释放SDA数据总线。
	_nop_();
}

/*检测从机应答*/
bit Test_ACK()
{
	SCL = 1;
	delay_5us();
	if (SDA)
	{
		SCL = 0;
		_nop_();
		I2C_Stop();
		return(0);
	}
	else
	{
		SCL = 0;
		_nop_();
		return(1);
	}
}


/*发送一个字节*/
void I2C_send_byte(uchar byte)
{
	uchar i;
	for(i = 0 ; i < 8 ; i++)
	{
		SCL = 0;
		_nop_();
		if (byte & 0x80)
		{				
			SDA = 1;	
			_nop_();				   
		}				
		else
		{
			SDA = 0;
			_nop_();
		}
		SCL = 1;
		_nop_();
		byte <<= 1;	// 0101 0100B 
	}
	SCL = 0;
	_nop_();
	SDA = 1;
	_nop_();
}


/*I2C 读一字节*/
uchar I2C_read_byte()
{
	uchar dat,i;
	SCL = 0;
	_nop_();
	SDA = 1;
	_nop_();
	for(i = 0 ; i < 8 ; i++)
	{
		SCL = 1;
		_nop_();
		if (SDA)			    
		{
			 dat |= 0x01; //
		}
		else
		{
			dat &=  0xfe;	//1111 1110
		}
		_nop_();
		SCL = 0 ;
		_nop_();
		if(i < 7)
		{
			dat = dat << 1;	
		}
	}
	return(dat);
}

uint *INT8UtostrHEX(uint num)	//将一个字节的数据转换为字符串
{
	uint i = 0;
	switch(num/100)
	{
		case 0:		strHEX[i] = '0'; 	i++;	break;
		case 1:		strHEX[i] = '1';	i++;	break;
		case 2:		strHEX[i] = '2';	i++;	break;
		case 3:		strHEX[i] = '3';	i++;	break;
		case 4:		strHEX[i] = '4';	i++;	break;
		case 5:		strHEX[i] = '5';	i++;	break;
		case 6:		strHEX[i] = '6';	i++;	break;
		case 7:		strHEX[i] = '7';	i++;	break;
		case 8:		strHEX[i] = '8';	i++;	break;
		case 9:		strHEX[i] = '9';	i++;	break;
	
		}

	switch(num%100/10)
	{
		case 0:		strHEX[i] = '0';	i++;	break;
		case 1:		strHEX[i] = '1';	i++;	break;
		case 2:		strHEX[i] = '2';	i++;	break;
		case 3:		strHEX[i] = '3';	i++;	break;
		case 4:		strHEX[i] = '4';	i++;	break;
		case 5:		strHEX[i] = '5';	i++;	break;
		case 6:		strHEX[i] = '6';	i++;	break;
		case 7:		strHEX[i] = '7';	i++;	break;
		case 8:		strHEX[i] = '8';	i++;	break;
		case 9:		strHEX[i] = '9';	i++;	break;
	
	}

		switch(num%100%10)
	{
		case 0:		strHEX[i] = '0';	i++;	break;
		case 1:		strHEX[i] = '1';	i++;	break;
		case 2:		strHEX[i] = '2';	i++;	break;
		case 3:		strHEX[i] = '3';	i++;	break;
		case 4:		strHEX[i] = '4';	i++;	break;
		case 5:		strHEX[i] = '5';	i++;	break;
		case 6:		strHEX[i] = '6';	i++;	break;
		case 7:		strHEX[i] = '7';	i++;	break;
		case 8:		strHEX[i] = '8';	i++;	break;
		case 9:		strHEX[i] = '9';	i++;	break;

		}
		
	strHEX[3] = '\0';
	return (strHEX);

	
}

void writeComm(uint comm)
{ 
 //  	while(LCD1602_Check_Busy()); //忙则等待
    LCD_RS  = 0;    
	P0=comm;
   LCDEN = 1;
    _nop_();
    LCDEN = 0;
	delay_ms(1);
}

//写数据
void writeData(uint dat)
{
 //  while(LCD1602_Check_Busy()); //忙则等待
     LCD_RS = 1; 
   P0 = dat;
	LCDEN = 1;
  _nop_();
    LCDEN = 0;
	delay_ms(1);
 }

//写字符串
void LCD1602_Write_String(uint x,uint y,uint *s) //LCD1602写字符串
{     
	if (y == 0) 
	{     
		writeComm(0x80 + x);     //表示第一行
	}
	else 
	{      
		writeComm(0xC0 + x);      //表示第二行
	}        
	while (*s != '\0') 
	{     
		writeData(*s++);         
	}
}

 //初始化显示屏
 void init()
 {
     LCD_RW = 0; 
     dula = wela = 0;
	 	LCDEN=0;
    writeComm(0x38);// 16*2显示 
   writeComm(0x0c); //打开显示 
    writeComm(0x06);//显示指针加1 
    writeComm(0x01); //清屏 
	 delay_ms(5);
}


/*ad读数据*/
bit I2C_ADC_ReadData(uchar ADDR, uint *ADC_Value)
{
	I2C_Start() ;
	I2C_send_byte(PCF8591_ADDR + 0);
	if (!Test_ACK())
	{
		return(0);
	}
	I2C_send_byte(ADDR);
	Master_ACK(0);
	I2C_Start();
	I2C_send_byte(PCF8591_ADDR + 1);
	if (!Test_ACK())
	{
		return(0);
	}
	*ADC_Value = I2C_read_byte();
	Master_ACK(0);
	I2C_Stop();
	return(1);	
}

void main()
{
	uint ADC_Value;
	init(); //LCD1602初始化
	I2C_init();
//  	LCD1602_Write_String(0, 0, "CH0:");
 
	while(1)
	{
	I2C_ADC_ReadData(3, &ADC_Value);	
	LCD1602_Write_String(0, 0,INT8UtostrHEX(ADC_Value));
	delay_ms(400); 
	}

}

  • 17
    点赞
  • 113
    收藏
    觉得还不错? 一键收藏
  • 12
    评论
yl69土壤湿度传感器是一种常用的土壤湿度检测传感器,可以用于测量土壤湿度水分含量。它通常由两个部分组成:一个是传感器模块,另一个是控制模块。 传感器模块包含两个针状电极,可以插入到土壤中。当土壤湿度较高时,土壤中的电导率也会相应增加,传感器模块会检测到这种变化,并输出一个电压信号。 控制模块通常使用Arduino或者树莓派等单片机来接收传感器模块输出的电压信号,并通过编程来处理和显示数据。在Python中,你可以使用相应的库来读取传感器模块输出的电压值,并将其转换为土壤湿度的百分比。 以下是一个简单的示例代码,演示如何使用yl69土壤湿度传感器和Python进行读取和显示: ```python import RPi.GPIO as GPIO import time # 设置GPIO引脚 channel = 17 GPIO.setmode(GPIO.BCM) GPIO.setup(channel, GPIO.IN) def callback(channel): if GPIO.input(channel): print("土壤湿度低") else: print("土壤湿度高") # 添加事件检测 GPIO.add_event_detect(channel, GPIO.BOTH, bouncetime=300) GPIO.add_event_callback(channel, callback) # 主循环 try: while True: time.sleep(1) except KeyboardInterrupt: GPIO.cleanup() ``` 这段代码使用了RPi.GPIO库来控制树莓派的GPIO引脚。它通过设置一个回调函数来检测传感器模块输出的电压变化,并根据电压值判断土壤湿度的高低。 你可以根据自己的需求修改代码,例如添加数据存储、显示等功能。同时,你也可以参考yl69土壤湿度传感器的相关文档和示例代码,以便更好地理解和使用它。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 12
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值