久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费
標題:
單片機智能溫室大棚DHT11溫濕度上下限報警控制源程序
[打印本頁]
作者:
lilei1219
時間:
2018-4-17 10:02
標題:
單片機智能溫室大棚DHT11溫濕度上下限報警控制源程序
用于智能溫室大棚環境監測的控制系統源代碼
一、程序用什么軟件查看?
打開程序的軟件是Keil,我們提供了相關的安裝包,放在資料包
《26、軟件--Keil 安裝包》里面。
二、如何復制到word?
1、進入“程序(Keil版)”這個文件夾;
2、找到“main.c”這個文件,程序源碼就在這個文件里面;
3、用電腦自帶的記事本軟件可以打開這個“main.c”文件;
4、用記事本打開之后,自己復制到word就可以了;
三、為什么程序中沒看到控制繼電器(風扇)的代碼?
繼電器和LED燈是連在一起的,因此控制了LED燈的同時,繼電器也控制了,具體原理參考資料包02原理圖分析中的繼電器一節。
四、如何下載程序到板子上?
我們提供了下載程序的視頻教程,放在資料包《20、如何下載單片機程序》。
單片機源程序如下:
#include <reg52.h> // 頭文件包含
#include <intrins.h>
#define uchar unsigned char // 以后unsigned char就可以用uchar代替
#define uint unsigned int // 以后unsigned int 就可以用uint 代替
sfr ISP_DATA = 0xe2; // 數據寄存器
sfr ISP_ADDRH = 0xe3; // 地址寄存器高八位
sfr ISP_ADDRL = 0xe4; // 地址寄存器低八位
sfr ISP_CMD = 0xe5; // 命令寄存器
sfr ISP_TRIG = 0xe6; // 命令觸發寄存器
sfr ISP_CONTR = 0xe7; // 命令寄存器
sbit LcdRs_P = P1^2; // 1602液晶的RS管腳
sbit LcdRw_P = P1^3; // 1602液晶的RW管腳
sbit LcdEn_P = P1^4; // 1602液晶的EN管腳
sbit Buzzer_P = P1^5; // 蜂鳴器
sbit DHT11_P = P1^6; // 溫濕度傳感器DHT11數據接入
sbit KeySet_P = P3^2; // “設置”按鍵的管腳
sbit KeyDown_P = P3^3; // “減”按鍵的管腳
sbit KeyUp_P = P3^4; // “加”按鍵的管腳
sbit LedTH_P = P2^0; // 溫度過高報警指示燈(同時控制降溫設備)
sbit LedTL_P = P2^1; // 溫度過低報警指示燈(同時控制加溫設備)
sbit LedHH_P = P2^2; // 濕度過高報警指示燈(同時控制除濕設備)
sbit LedHL_P = P2^3; // 濕度過低報警指示燈(同時控制加濕設備)
uchar temp; // 保存溫度
uchar humi; // 保存濕度
uchar AlarmTL; // 溫度下限報警值
uchar AlarmTH; // 溫度上限報警值
uchar AlarmHL; // 濕度下限報警值
uchar AlarmHH; // 濕度上限報警值
/*********************************************************/
// 單片機內部EEPROM不使能
/*********************************************************/
void ISP_Disable()
{
ISP_CONTR = 0;
ISP_ADDRH = 0;
ISP_ADDRL = 0;
}
/*********************************************************/
// 從單片機內部EEPROM讀一個字節,從0x2000地址開始
/*********************************************************/
unsigned char EEPROM_Read(unsigned int add)
{
ISP_DATA = 0x00;
ISP_CONTR = 0x83;
ISP_CMD = 0x01;
ISP_ADDRH = (unsigned char)(add>>8);
ISP_ADDRL = (unsigned char)(add&0xff);
// 對STC89C51系列來說,每次要寫入0x46,再寫入0xB9,ISP/IAP才會生效
ISP_TRIG = 0x46;
ISP_TRIG = 0xB9;
_nop_();
ISP_Disable();
return (ISP_DATA);
}
/*********************************************************/
// 往單片機內部EEPROM寫一個字節,從0x2000地址開始
/*********************************************************/
void EEPROM_Write(unsigned int add,unsigned char ch)
{
ISP_CONTR = 0x83;
ISP_CMD = 0x02;
ISP_ADDRH = (unsigned char)(add>>8);
ISP_ADDRL = (unsigned char)(add&0xff);
ISP_DATA = ch;
ISP_TRIG = 0x46;
ISP_TRIG = 0xB9;
_nop_();
ISP_Disable();
}
/*********************************************************/
// 擦除單片機內部EEPROM的一個扇區
// 寫8個扇區中隨便一個的地址,便擦除該扇區,寫入前要先擦除
/*********************************************************/
void Sector_Erase(unsigned int add)
{
ISP_CONTR = 0x83;
ISP_CMD = 0x03;
ISP_ADDRH = (unsigned char)(add>>8);
ISP_ADDRL = (unsigned char)(add&0xff);
ISP_TRIG = 0x46;
ISP_TRIG = 0xB9;
_nop_();
ISP_Disable();
}
/*********************************************************/
// 毫秒級的延時函數,time是要延時的毫秒數
/*********************************************************/
void DelayMs(uint time)
{
uint i,j;
for(i=0;i<time;i++)
for(j=0;j<112;j++);
}
/*********************************************************/
// 1602液晶寫命令函數,cmd就是要寫入的命令
/*********************************************************/
void LcdWriteCmd(uchar cmd)
{
LcdRs_P = 0;
LcdRw_P = 0;
LcdEn_P = 0;
P0=cmd;
DelayMs(2);
LcdEn_P = 1;
DelayMs(2);
LcdEn_P = 0;
}
/*********************************************************/
// 1602液晶寫數據函數,dat就是要寫入的數據
/*********************************************************/
void LcdWriteData(uchar dat)
{
LcdRs_P = 1;
LcdRw_P = 0;
LcdEn_P = 0;
P0=dat;
DelayMs(2);
LcdEn_P = 1;
DelayMs(2);
LcdEn_P = 0;
}
/*********************************************************/
// 1602液晶初始化函數
/*********************************************************/
void LcdInit()
{
LcdWriteCmd(0x38); // 16*2顯示,5*7點陣,8位數據口
LcdWriteCmd(0x0C); // 開顯示,不顯示光標
LcdWriteCmd(0x06); // 地址加1,當寫入數據后光標右移
LcdWriteCmd(0x01); // 清屏
}
/*********************************************************/
// 液晶光標定位函數
/*********************************************************/
void LcdGotoXY(uchar line,uchar column)
{
// 第一行
if(line==0)
LcdWriteCmd(0x80+column);
// 第二行
if(line==1)
LcdWriteCmd(0x80+0x40+column);
}
/*********************************************************/
// 液晶輸出字符串函數
/*********************************************************/
void LcdPrintStr(uchar *str)
{
while(*str!='\0') // 判斷是否到字符串的盡頭了
LcdWriteData(*str++);
}
/*********************************************************/
// 液晶輸出數字
/*********************************************************/
void LcdPrintNum(uchar num)
{
LcdWriteData(num/10+48); // 十位
LcdWriteData(num%10+48); // 個位
}
/*********************************************************/
// 液晶顯示內容的初始化
/*********************************************************/
void LcdShowInit()
{
LcdGotoXY(0,0); // 第0行的顯示內容
LcdPrintStr(" DHT11 System ");
LcdGotoXY(1,0); // 第1行的顯示內容
LcdPrintStr("T: C H: %RH");
LcdGotoXY(1,4); // 溫度單位攝氏度上面的圓圈符號
LcdWriteData(0xdf);
}
/*********************************************************/
// 10us級延時程序
/*********************************************************/
void Delay10us()
{
_nop_(); // 執行一條指令,延時1微秒
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
}
/*********************************************************/
// 讀取DHT11單總線上的一個字節
/*********************************************************/
uchar DhtReadByte(void)
{
bit bit_i;
uchar j;
uchar dat=0;
for(j=0;j<8;j++)
{
while(!DHT11_P); // 等待低電平結束
Delay10us(); // 延時
Delay10us();
Delay10us();
if(DHT11_P==1) // 判斷數據線是高電平還是低電平
{
bit_i=1;
while(DHT11_P);
}
else
{
bit_i=0;
}
dat<<=1; // 將該位移位保存到dat變量中
dat|=bit_i;
}
return(dat);
}
/*********************************************************/
// 讀取DHT11的一幀數據,濕高、濕低(0)、溫高、溫低(0)、校驗碼
/*********************************************************/
void ReadDhtData()
{
uchar HumiHig; // 濕度高檢測值
uchar HumiLow; // 濕度低檢測值
uchar TemHig; // 溫度高檢測值
uchar TemLow; // 溫度低檢測值
uchar check; // 校驗字節
DHT11_P=0; // 主機拉低
DelayMs(20); // 保持20毫秒
DHT11_P=1; // DATA總線由上拉電阻拉高
Delay10us(); // 延時等待30us
Delay10us();
Delay10us();
while(!DHT11_P); // 等待DHT的低電平結束
while(DHT11_P); // 等待DHT的高電平結束
//進入數據接收狀態
HumiHig = DhtReadByte(); // 濕度高8位
HumiLow = DhtReadByte(); // 濕度低8為,總為0
TemHig = DhtReadByte(); // 溫度高8位
TemLow = DhtReadByte(); // 溫度低8為,總為0
check = DhtReadByte(); // 8位校驗碼,其值等于讀出的四個字節相加之和的低8位
DHT11_P=1; // 拉高總線
if(check==HumiHig + HumiLow + TemHig + TemLow) // 如果收到的數據無誤
{
temp=TemHig; // 將溫度的檢測結果賦值給全局變量temp
humi=HumiHig; // 將濕度的檢測結果賦值給全局變量humi
}
}
/*********************************************************/
// 是否需要報警判斷
/*********************************************************/
void AlarmJudge(void)
{
uchar i;
if(temp>AlarmTH) // 溫度是否過高
{
LedTH_P=0;
LedTL_P=1;
}
else if(temp<AlarmTL) // 溫度是否過低
{
LedTL_P=0;
LedTH_P=1;
}
else // 溫度正常
{
LedTH_P=1;
LedTL_P=1;
}
if(humi>AlarmHH) // 濕度是否過高
{
LedHH_P=0;
LedHL_P=1;
}
else if(humi<AlarmHL) // 濕度是否過低
{
LedHL_P=0;
LedHH_P=1;
}
else // 濕度正常
{
LedHH_P=1;
LedHL_P=1;
}
if((LedHH_P==0)||(LedHL_P==0)||(LedTH_P==0)||(LedTL_P==0)) // 蜂鳴器判斷,只要至少1個報警燈亮,蜂鳴器就報警
{
for(i=0;i<3;i++)
{
Buzzer_P=0;
DelayMs(100);
Buzzer_P=1;
DelayMs(100);
}
}
}
/*********************************************************/
// 按鍵掃描,用于設置溫濕度報警范圍
/*********************************************************/
void KeyScanf()
{
if(KeySet_P==0) // 判斷設置按鍵是否被按下
{
/*將液晶顯示改為設置頁面的*******************************************************/
LcdWriteCmd(0x01); // 設置界面的顯示框架
LcdGotoXY(0,0);
LcdPrintStr("Temp: - ");
LcdGotoXY(1,0);
LcdPrintStr("Humi: - ");
LcdGotoXY(0,6); // 在液晶上填充溫度的下限值
LcdPrintNum(AlarmTL);
LcdGotoXY(0,9); // 在液晶上填充溫度的上限值
LcdPrintNum(AlarmTH);
LcdGotoXY(1,6); // 在液晶上填充濕度的下限值
LcdPrintNum(AlarmHL);
LcdGotoXY(1,9); // 在液晶上填充濕度的上限值
LcdPrintNum(AlarmHH);
LcdGotoXY(0,7); // 光標定位到第0行第7列
LcdWriteCmd(0x0F); // 光標閃爍
DelayMs(10); // 去除按鍵按下的抖動
while(!KeySet_P); // 等待按鍵釋放
DelayMs(10); // 去除按鍵松開的抖動
/*設置溫度的下限值****************************************************************/
while(KeySet_P) // “設置鍵”沒有被按下,則一直處于溫度下限的設置
{
if(KeyDown_P==0) // 判斷 “減按鍵“ 是否被按下
{
if(AlarmTL>0) // 只有當溫度下限值大于0時,才能減1
AlarmTL--;
LcdGotoXY(0,6); // 重新刷新顯示更改后的溫度下限值
LcdPrintNum(AlarmTL);
LcdGotoXY(0,7); // 重新定位閃爍的光標位置
DelayMs(350); // 延時
}
if(KeyUp_P==0) // 判斷 “加按鍵“ 是否被按下
{
if(AlarmTL<99) // 只有當溫度下限值小于99時,才能加1
AlarmTL++;
LcdGotoXY(0,6); // 重新刷新顯示更改后的溫度下限值
LcdPrintNum(AlarmTL);
LcdGotoXY(0,7); // 重新定位閃爍的光標位置
DelayMs(350); // 延時
}
}
LcdGotoXY(0,10);
DelayMs(10); // 去除按鍵按下的抖動
while(!KeySet_P); // 等待按鍵釋放
DelayMs(10); // 去除按鍵松開的抖動
/*設置溫度的上限值****************************************************************/
while(KeySet_P) // “設置鍵”沒有被按下,則一直處于溫度上限的設置
{
if(KeyDown_P==0) // 判斷 “減按鍵“ 是否被按下
{
if(AlarmTH>0) // 只有當溫度上限值大于0時,才能減1
AlarmTH--;
LcdGotoXY(0,9); // 重新刷新顯示更改后的溫度上限值
LcdPrintNum(AlarmTH);
LcdGotoXY(0,10); // 重新定位閃爍的光標位置
DelayMs(350); // 延時
}
if(KeyUp_P==0) // 判斷 “加按鍵“ 是否被按下
{
if(AlarmTH<99) // 只有當溫度上限值小于99時,才能加1
AlarmTH++;
LcdGotoXY(0,9); // 重新刷新顯示更改后的溫度上限值
LcdPrintNum(AlarmTH);
LcdGotoXY(0,10); // 重新定位閃爍的光標位置
DelayMs(350); // 延時
}
}
LcdGotoXY(1,7);
DelayMs(10); // 去除按鍵按下的抖動
while(!KeySet_P); // 等待按鍵釋放
DelayMs(10); // 去除按鍵松開的抖動
/*設置濕度的下限值****************************************************************/
while(KeySet_P) // “設置鍵”沒有被按下,則一直處于濕度下限的設置
{
if(KeyDown_P==0) // 判斷 “減按鍵“ 是否被按下
{
if(AlarmHL>0) // 只有當濕度下限值大于0時,才能減1
AlarmHL--;
LcdGotoXY(1,6); // 重新刷新顯示更改后的濕度下限值
LcdPrintNum(AlarmHL);
LcdGotoXY(1,7); // 重新定位閃爍的光標位置
DelayMs(350);
}
if(KeyUp_P==0) // 判斷 “加按鍵“ 是否被按下
{
if(AlarmHL<99) // 只有當濕度下限值小于99時,才能加1
AlarmHL++;
LcdGotoXY(1,6); // 重新刷新顯示更改后的濕度下限值
LcdPrintNum(AlarmHL);
LcdGotoXY(1,7); // 重新定位閃爍的光標位置
DelayMs(350); // 延時
}
}
LcdGotoXY(1,10);
DelayMs(10); // 去除按鍵按下的抖動
while(!KeySet_P); // 等待按鍵釋放
DelayMs(10); // 去除按鍵松開的抖動
/*設置濕度的上限值****************************************************************/
while(KeySet_P) // “設置鍵”沒有被按下,則一直處于濕度上限的設置
{
if(KeyDown_P==0) // 判斷 “減按鍵“ 是否被按下
{
if(AlarmHH>0) // 只有當濕度上限值大于0時,才能減1
AlarmHH--;
LcdGotoXY(1,9); // 重新刷新顯示更改后的濕度上限值
LcdPrintNum(AlarmHH);
LcdGotoXY(1,10); // 重新定位閃爍的光標位置
DelayMs(350);
}
if(KeyUp_P==0) // 判斷 “加按鍵“ 是否被按下
{
if(AlarmHH<99) // 只有當濕度上限值小于99時,才能加1
AlarmHH++;
LcdGotoXY(1,9); // 重新刷新顯示更改后的濕度上限值
LcdPrintNum(AlarmHH);
LcdGotoXY(1,10); // 重新定位閃爍的光標位置
DelayMs(350); // 延時
}
}
LcdWriteCmd(0x0C); // 取消光標閃爍
LcdShowInit(); // 液晶顯示為檢測界面的
DelayMs(10); // 去除按鍵按下的抖動
while(!KeySet_P); // 等待按鍵釋放
DelayMs(10); // 去除按鍵松開的抖動
Sector_Erase(0x2000); // 存儲之前必須先擦除
EEPROM_Write(0x2000,AlarmTL); // 把溫度下限存入到EEPROM的0x2000這個地址
EEPROM_Write(0x2001,AlarmTH); // 把溫度上限存入到EEPROM的0x2001這個地址
EEPROM_Write(0x2002,AlarmHL); // 把濕度下限存入到EEPROM的0x2002這個地址
EEPROM_Write(0x2003,AlarmHH); // 把濕度上限存入到EEPROM的0x2003這個地址
}
}
/*********************************************************/
// 主函數
/*********************************************************/
void main()
{
uchar i;
LcdInit(); // 液晶功能的初始化
LcdShowInit(); // 液晶顯示的初始化
AlarmTL=EEPROM_Read(0x2000); // 從EEPROM的0x2000這個地址讀取溫度的報警下限
AlarmTH=EEPROM_Read(0x2001); // 從EEPROM的0x2001這個地址讀取溫度的報警上限
AlarmHL=EEPROM_Read(0x2002); // 從EEPROM的0x2002這個地址讀取濕度的報警下限
AlarmHH=EEPROM_Read(0x2003); // 從EEPROM的0x2003這個地址讀取濕度的報警上限
if((AlarmTL==0)||(AlarmTL>100)) // 如果溫度下限報警值讀出來異常(等于0或大于100),則重新賦值
AlarmTL=20;
if((AlarmTH==0)||(AlarmTH>100)) // 如果溫度上限報警值讀出來異常(等于0或大于100),則重新賦值
AlarmTH=35;
if((AlarmHL==0)||(AlarmHL>100)) // 如果溫度下限報警值讀出來異常(等于0或大于100),則重新賦值
AlarmHL=40;
if((AlarmHH==0)||(AlarmHH>100)) // 如果溫度上限報警值讀出來異常(等于0或大于100),則重新賦值
……………………
…………限于本文篇幅 余下代碼請從51黑下載附件…………
復制代碼
所有資料51hei提供下載:
07、單片機程序.rar
(43.89 KB, 下載次數: 238)
2018-4-17 10:01 上傳
點擊文件名下載附件
源代碼
下載積分: 黑幣 -5
作者:
aoyucheng
時間:
2018-4-25 15:26
很煩嘛,我的是顯示溫濕度和時間日前的 lcd1602都顯示不夠
作者:
chb333
時間:
2018-5-7 10:16
有仿真圖這些嗎
作者:
不說
時間:
2021-2-24 15:20
請問有原理圖嗎
歡迎光臨 (http://www.zg4o1577.cn/bbs/)
Powered by Discuz! X3.1
主站蜘蛛池模板:
国产精品久久99
|
亚洲一区二区精品视频
|
91视视频在线观看入口直接观看
|
99re在线视频
|
亚洲视频在线免费
|
japanhdxxxx裸体
|
亚洲一二三在线观看
|
欧美一级片在线观看
|
中文字幕亚洲欧美
|
精品真实国产乱文在线
|
中文字幕日韩欧美一区二区三区
|
亚洲精品电影在线观看
|
国产成人精品午夜视频免费
|
丝袜天堂
|
成人h片在线观看
|
国产成人精品一区二区三区网站观看
|
国产视频第一页
|
羞羞的视频免费在线观看
|
蜜桃精品在线
|
日韩黄色av
|
亚洲91视频
|
国产传媒
|
91亚洲国产成人久久精品网站
|
99久久精品免费看国产免费软件
|
中文字幕在线电影观看
|
欧美精品一区二区三区四区
|
国产视频福利在线观看
|
午夜a√
|
国产成人免费视频网站视频社区
|
亚洲精品中文字幕中文字幕
|
国产午夜精品久久久
|
亚洲一区二区日韩
|
影音先锋欧美资源
|
国产高清在线视频
|
亚洲综合大片69999
|
久久久av
|
毛色毛片免费看
|
亚洲成人在线免费
|
亚洲精品日韩欧美
|
99久久国产综合精品麻豆
|
在线观看免费福利
|