基於51單片機的智能溫控風扇

1.功能
本設計爲一種溫控風扇系統,具有靈敏的溫度感測和顯示功能,系統選用STC89C52單片機作爲控制平臺對風扇轉速進行控制。可在測得溫度值在高低溫度之間時打開風扇弱風檔,當溫度升高超過所設定的溫度時自動切換到大風檔,當溫度小於所設定的溫度時自動關閉風扇,控制狀態隨外界溫度而定。

2.硬件設計
在這裏插入圖片描述
硬件電路主要由:

  1. 單片機最小系統
  2. 風扇驅動電路
  3. LCD1602顯示屏電路
  4. DS18B20溫度採集電路

3.程序設計

(1)LCD1602驅動程序

#define LCD1602_DB  P0
sbit LCD1602_RS = P2^0;
sbit LCD1602_RW = P2^1;
sbit LCD1602_E  = P2^2;

/* 等待液晶準備好 */
void LcdWaitReady()
{
    unsigned char sta;
    
    LCD1602_DB = 0xFF;
    LCD1602_RS = 0;
    LCD1602_RW = 1;
    do {
        LCD1602_E = 1;
        sta = LCD1602_DB; //讀取狀態字
        LCD1602_E = 0;
    } while (sta & 0x80); //bit7等於1表示液晶正忙,重複檢測直到其等於0爲止
}
/* 向LCD1602液晶寫入一字節命令,cmd-待寫入命令值 */
void LcdWriteCmd(unsigned char cmd)
{
    LcdWaitReady();
    LCD1602_RS = 0;
    LCD1602_RW = 0;
    LCD1602_DB = cmd;
    LCD1602_E  = 1;
    LCD1602_E  = 0;
}
/* 向LCD1602液晶寫入一字節數據,dat-待寫入數據值 */
void LcdWriteDat(unsigned char dat)
{
    LcdWaitReady();
    LCD1602_RS = 1;
    LCD1602_RW = 0;
    LCD1602_DB = dat;
    LCD1602_E  = 1;
    LCD1602_E  = 0;
}
/* 設置顯示RAM起始地址,亦即光標位置,(x,y)-對應屏幕上的字符座標 */
void LcdSetCursor(unsigned char x, unsigned char y)
{
    unsigned char addr;
    
    if (y == 0)  //由輸入的屏幕座標計算顯示RAM的地址
        addr = 0x00 + x;  //第一行字符地址從0x00起始
    else
        addr = 0x40 + x;  //第二行字符地址從0x40起始
    LcdWriteCmd(addr | 0x80);  //設置RAM地址
}
/* 在液晶上顯示字符串,(x,y)-對應屏幕上的起始座標,str-字符串指針 */
void LcdShowStr(unsigned char x, unsigned char y, unsigned char *str)
{
    LcdSetCursor(x, y);   //設置起始地址
    while (*str != '\0')  //連續寫入字符串數據,直到檢測到結束符
    {
        LcdWriteDat(*str++);
    }
}
/* 初始化1602液晶 */
void InitLcd1602()
{
    LcdWriteCmd(0x38);  //16*2顯示,5*7點陣,8位數據接口
    LcdWriteCmd(0x0C);  //顯示器開,光標關閉
    LcdWriteCmd(0x06);  //文字不動,地址自動+1
    LcdWriteCmd(0x01);  //清屏
}

(2)DS18B20驅動程序

sbit IO_18B20=P3^2;

/*軟件延時函數,延時時間(t*10)us*/
void DelayX10us(unsigned char t)
{
	do{
		_nop_();
		_nop_();
		_nop_();
		_nop_();
		_nop_();
		_nop_();
		_nop_();
		_nop_();
	}while(--t);
}
/*復位總線,獲取存在脈衝,以啓動一次讀寫操作*/
bit Get18B20Ack()
{
	bit ack;
	
	EA=0;	   			//禁止總中斷
	IO_18B20=0;			//產生500us復位脈衝
	DelayX10us(50);
	IO_18B20=1;
	DelayX10us(6);		//延時60us
	ack=IO_18B20;		//讀取存在脈衝
	while(!IO_18B20);   //等待存在脈衝結束
	EA=1;				//重新使能總中斷
	return ack;
}
/*向DS18B20寫入一個字節,dat-待寫入字節*/
void Write18B20(unsigned char dat)
{
	unsigned char mask;
	EA=0;
	for(mask=0x01;mask!=0;mask<<=1)//低位在先,依次移出8個bit
	{
		IO_18B20=0;//產生2us低電平脈衝
		_nop_();
		_nop_();
		if((mask&dat)==0)//輸出該bit值
			IO_18B20=0;
		else
			IO_18B20=1;
		DelayX10us(6);//延時60us
		IO_18B20=1;//拉高通信引腳
	}
	EA=1;
}
/*從DS18B20讀取一個字節,返回值-讀到的字節*/
unsigned char Read18B20()
{
	unsigned char dat;
	unsigned char mask;

	EA=0;
	for(mask=0x01;mask!=0;mask<<=1)//低位在先,依次採集8個bit
	{
		IO_18B20=0;//產生2us低電平脈衝
		_nop_();
		_nop_();
		IO_18B20=1;//結束低電平脈衝,等待18B20輸出數據
		_nop_(); //延時2us
		_nop_();
		if(!IO_18B20)//讀取通信引腳上的值
			dat &= ~mask;
		else
			dat |= mask;
		DelayX10us(6);//再延時60us
	}
	EA=1;
	return dat;
}
/*啓動一次18B20溫度轉換,返回值-表示是否啓動成功*/
bit Start18B20()
{
	bit ack;
	ack=Get18B20Ack();//執行總線復位,並獲取18B20應答
	if(ack==0)
	{
		Write18B20(0xCC);
		Write18B20(0x44);
	}
	return ~ack;
}
/*讀取DS18B20轉換的溫度值,返回值-表示是否讀取成功*/
bit Get18B20Temp(int *temp)
{
	bit ack;
	unsigned char LSB,MSB;//16bit溫度值的低字節和高字節

	ack=Get18B20Ack();//執行總線復位,並獲取18B20應答
	if(ack==0)
	{
		Write18B20(0xCC);//跳過ROM操作
		Write18B20(0xBE);//發送讀命令
		LSB=Read18B20();//讀溫度值的低字節
		MSB=Read18B20();//讀溫度值的高字節
		*temp=((int)MSB<<8)+LSB;//合成爲16bit整型數
	}
	return ~ack;
}

(3)主程序

sbit IN1=P2^7;
sbit IN2=P2^6;
sbit ENA=P2^5;

bit flag1s=0;//1s定時標誌
unsigned char T0RH=0;
unsigned char T0RL=0;

int temp;//讀取到的當前溫度值
unsigned char len;
int intT,decT;//溫度值的整數和小數部分
unsigned char str[12];

void Compare();
void GetTemp();
void ConfigTimer0(unsigned int ms);
unsigned char IntToString(unsigned char *str,int dat);
extern bit Start18B20();
extern bit Get18B20Temp(int *temp);
extern void InitLcd1602();
extern void LcdShowStr(unsigned char x,unsigned char y,unsigned char *str);

void main()
{
	bit res;

	EA=1;
	ConfigTimer0(10);//T0定時10ms
	Start18B20();//啓動DS18B20
	InitLcd1602();//初始化液晶

	while(1)
	{		
		if(flag1s)//每秒更新一次溫度
		{
			flag1s=0;
			res=Get18B20Temp(&temp);//讀取當前溫度
			if(res)//讀取成功時,刷新當前溫度顯示
			{
				GetTemp();
			
				LcdShowStr(0,0,"Welcome to use");//顯示字符及溫度值
				LcdShowStr(0,1,"Current T:");
				LcdShowStr(10,1,str);
					Compare();
			}
			else //讀取失敗時,提示錯誤信息
			{
				LcdShowStr(0,0,"error!");

			}
			Start18B20();//重新啓動下一次轉換					 
		}
	}
}
/*溫度獲取函數,獲取當前環境溫度值並保存在str數組中*/
void GetTemp()
{

	intT=temp>>4;//分離出溫度值整數部分
	decT=temp &0x0F;//分離出溫度值小數部分
			
	len=IntToString(str,intT);//整數部分轉換成字符串
			
	str[len++]='.';
	decT=(decT*10)/16;//二進制的小數部分轉換爲1位十進制位
	str[len++]=decT+'0';//十進制小數位再轉換爲ASCII字符
	while(len<6)//用空格補齊到6個字符長度
	{
		str[len++]=' ';
	}
	str[len++]='\0';
}
/*延時函數,用於PWM控制*/
void delay(unsigned int z)
{
	unsigned int x,y;
	for(x=z;x>0;x--)
		for(y=110;y>0;y--);
} 
/*比較函數,通過溫度值的比較設置電機的轉速*/
void Compare()
{
	unsigned int i=0;
	unsigned char j;

	if((intT>= 24) && (intT<26))   //以兩度爲一個溫差範圍,並設溫度範圍索引
	{
		j=0;	
	}
	else if((intT>=26) &&(intT<28))
	{
		j=1;
	}
	else if((intT>=28) &&(intT<30))
	{
		j=2;
	}
	else if(intT>=30)
	{
		j=3;
	}
	switch(j)		  //根據溫度索引設置電機轉速
	{
		case 0:	IN1=1;
				IN2=0;
		  		for(i=0;i<200;i++)
	      		{
					ENA=1;
	     			delay(20);
	      		    ENA=0;
					delay(30);
				}
				break;
	
		case 1:	IN1=1;
				IN2=0;
		  		for(i=0;i<200;i++)
	      		{
					ENA=1;
	     			delay(30);
	      		    ENA=0;
					delay(30);
				}
				break;	 
	
		case 2:	IN1=1;
				IN2=0;
		  		for(i=0;i<200;i++)
	      		{
					ENA=1;
	     			delay(55);			 
	      		    ENA=0;
					delay(30);
				}
				break;	 
							
		case 3:	IN1=1;
				IN2=0;
		  	    ENA=1;
				break;

		default:break;	 	 
	}
}  

/*整型數轉換爲字符串,str-字符串指針,dat-待轉換數,返回值-字符串長度*/
unsigned char IntToString(unsigned char *str,int dat)
{
	signed char i=0;
	unsigned char len=0;
	unsigned char buf[6];

	if(dat<0)//如果爲負數,首先取絕對值,並在指針上添加負號
	{
	 	dat=-dat;
		*str++='-';
		len++;
	}
	do{	   //先轉換爲低位在前的十進制數組
		buf[i++]=dat%10;
		dat /=10;
	}while(dat>0);
	len += i;//i最後的值就是有效字符的個數
	while(i-->0)//將數組值轉換爲ASCII碼反向拷貝到接收指針上
	{
		*str++=buf[i]+'0';
	}
	*str='\0';
	return len;
}
void ConfigTimer0(unsigned int ms)
{
	unsigned long tmp;

	tmp=11059200/12;
	tmp=(tmp*ms)/1000;
	tmp=65536-tmp;
	tmp=tmp+12;
	T0RH=(unsigned char)(tmp>>8);
	T0RL=(unsigned char)tmp;
	TMOD &= 0xF0;
	TMOD |= 0x01;
	TH0=T0RH;
	TL0=T0RL;
	ET0=1;
	TR0=1;
}
void InterruptTimer0() interrupt 1
{
	static unsigned char tmr1s=0;

	TH0=T0RH;
	TL0=T0RL;
	tmr1s++;
	if(tmr1s>=100)
	{
		tmr1s=0;
		flag1s=1;
	}		 

}

源碼+電路圖 下載:關注公衆號,首頁回覆“溫控風扇”獲取資料
在這裏插入圖片描述

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章