单片机论坛

 找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 27444|回复: 106
收起左侧

单片机自动浇花系统(附原理图、仿真文件、源程序)

  [复制链接]
dai@1228 发表于 2017-12-1 12:02 | 显示全部楼层 |阅读模式
(共享资料)自己做的基于stc89c51单片机的自动浇花系统,有自动和手动两种模式,可以设置湿度的上下限,利用lcd1602显示,有一点遗憾的是设置湿度上下限时,没有光标闪动提示,希望大家给点建议,帮助改进,我把源程序附在这。
仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)

protues仿真

protues仿真


原理图:

原理图

原理图


单片机源代码:
  1. #include<reg51.h>
  2. #define uchar unsigned char
  3. #define uint unsigned int

  4. sbit ad_cs=P1^3;         //ADC0832的控制位
  5. sbit ad_clk=P1^0;
  6. sbit ad_dat=P1^1;

  7. sbit lcd_rs=P2^7;         //LCD1602的控制位
  8. sbit lcd_e=P2^6;

  9. sbit key1=P1^4;                 //独立按键
  10. sbit key2=P3^2;
  11. sbit key3=P3^3;

  12. sbit relay=P1^6;     //继电器

  13. bit flag=1;                     //工作模式标志
  14. bit flag_motor=1;         //手动模式下,抽水机开关标志
  15. bit flag_jia=0;
  16. bit flag_jian=0;

  17. uchar Hum_H=60;                 //自动模式下,湿度设定最大值
  18. uchar Hum_L=30;                 //自动模式下,湿度设定最小值

  19. uchar num=0;

  20. /********************延时函数1ms*****************
  21. void delay_ms(uint a)                                       
  22. {
  23.         uint i;
  24.         while(a--)
  25.         for(i=0;i<125;i++);
  26. }
  27. *************************************/


  28. /*******************延时函数50us******************/
  29. void delay_50us(uint a)
  30. {
  31.         uint i;
  32.         while(a--)
  33.         for(i=0;i<19;i++);
  34. }
  35. /*************************************/

  36. /*******************ADC读取******************/
  37. uchar read_ad(uchar channel)  
  38. {
  39.         uchar i;
  40.         uchar dat1,dat2;
  41.         ad_clk=0;
  42.         ad_cs=0;
  43.         ad_dat=1;
  44.         ad_clk=1;
  45.         ad_clk=0;  //起始信号

  46.         ad_dat=1;
  47.         ad_clk=1;
  48.         ad_clk=0;  //选择单通道模式

  49.         ad_dat=channel;//0对应通道0,1对应通道1
  50.         ad_clk=1;
  51.         ad_clk=0;

  52.         ad_dat=1;

  53.         for(i=0;i<8;++i)
  54.         {
  55.                  ad_clk=1;
  56.                 ad_clk=0;
  57.                 dat1=dat1<<1;
  58.                 if(ad_dat)
  59.                 dat1|=0x01;        
  60.         
  61.         }
  62.         for(i=0;i<8;i++)
  63.         {
  64.                  dat2=dat2>>1;
  65.                 if(ad_dat)
  66.                 dat2|=0x80;        
  67.                
  68.                 ad_clk=1;
  69.                 ad_clk=0;
  70.         }
  71.         ad_cs=1;
  72.         return (dat1==dat2)?dat1:0;
  73.         
  74. }
  75. /*************************************/

  76. /*******************LCD1602初始化******************/
  77. void write_com(uchar com)        //写指令
  78. {
  79.         lcd_e=0;
  80.         lcd_rs=0;
  81.         P0=com;
  82.         delay_50us(10);
  83.         lcd_e=1;
  84.         delay_50us(10);
  85.         lcd_e=0;
  86. }

  87. void write_data(uchar dat)  //写数据
  88. {
  89.         lcd_e=0;
  90.         lcd_rs=1;
  91.         P0=dat;
  92.         delay_50us(10);
  93.         lcd_e=1;
  94.         delay_50us(10);
  95.         lcd_e=0;
  96. }                        

  97. void init_lcd1602()                 //lcd 初始化
  98. {
  99.         delay_50us(300);
  100.         write_com(0x38);
  101.         delay_50us(100);
  102.         write_com(0x38);
  103.         delay_50us(100);
  104.         write_com(0x38);
  105.         write_com(0x38);
  106.         write_com(0x08);
  107.         write_com(0x01);
  108.         write_com(0x06);
  109.         write_com(0x0c);
  110. }

  111. void display_shu(uchar add,uchar dat)          //显示数字
  112. {
  113.         uchar l,m;
  114.         
  115. //        k=dat/100;
  116.         l=dat%100/10;
  117.         m=dat%10;

  118.         write_com(0x80+add);
  119. //        write_data(k+0x30);
  120.         write_data(l+0x30);
  121.         write_data(m+0x30);
  122. }

  123. void display_string(uchar add,uchar *dat)  //显示字符串
  124. {
  125.         uchar i;
  126.          write_com(0x80+add);
  127.         while(dat!=0&&(*dat!='\0')&&i<16)
  128.         {
  129.                  write_data(*dat);
  130.                 dat++;
  131.                 i++;
  132.                 if(i==15) i=0;
  133.         }
  134. }
  135. /*****************************************/

  136. /****************主函数*******************/
  137. void main()
  138. {
  139.         EA=1;                                                          //开总中断
  140.         EX0=1;                                                          //开外部中断0
  141.         EX1=1;                                                          //开外部中断1
  142.         IT0=0;                                                          //外部中断0触发方式为低电平
  143.         IT1=0;                                                          //外部中断1触发方式为低电平
  144.         
  145.         init_lcd1602();        
  146.         
  147.         display_string(0,"Hum:  %");      //实时显示当前的湿度
  148.         display_string(11,"H:  %");       //显示设置的湿度最大值

  149.         display_string(0x40,"Mode:");          //显示工作模式
  150.         display_string(0x40+11,"L:  %");  //显示设置的湿度最小值


  151.     while(1)
  152.         {        
  153.                 uchar i;
  154.                
  155.                 i=100-read_ad(0)*0.39;
  156.                 display_shu(4,i);
  157.                 display_shu(13,Hum_H);
  158.                 display_shu(0x40+13,Hum_L);
  159.                 if(key1==0)        
  160.                         {
  161.                                 delay_50us(100);
  162.                                 if(key1==0)
  163.                                 {
  164.                                         while(!key1);
  165.                                         num++;
  166.                                         if(num>2) num=0;
  167.                                 }        
  168.                            
  169.                                 switch (num)                                         //选择工作模式和调整湿度设置最大、最小值
  170.                             {
  171.                                 case 0:flag=~flag;
  172.                                            flag_jia=0;
  173.                                            flag_jian=0;
  174.                                            //write_com(0x80+0x40);
  175.                                        //write_com(0x0f);
  176.                                            break;
  177.                                 case 1://write_com(0x80+9);
  178.                                        //write_com(0x0f);
  179.                                            //delay_50us(1000);
  180.                                            //write_com(0x0c);
  181.                                            flag_jia=1;
  182.                                            flag_jian=0;
  183.                                            break;
  184.                                 case 2://write_com(0x80+0x40+9);
  185.                                        //write_com(0x0f);
  186.                                            //delay_50us(1000);
  187.                                            //write_com(0x0c);
  188.                                            flag_jian=1;
  189.                                            flag_jia=0;
  190.                                            break;        
  191.                            }
  192.                            if(flag==0&&num==1)        
  193.                            {
  194.                                    flag=~flag;
  195.                                    num=0;
  196.                                    flag_jia=0;
  197.                                    flag_jian=0;
  198.                            }               
  199.                     }                  
  200.                 if(flag==0)                              //手动模式
  201.                 {
  202.                         
  203.                          display_string(0x40+5,"MT");
  204.                         if(flag_motor) relay=1;
  205.                         else if(i<Hum_H) relay=0;          //不能超过湿度最大值
  206.                         else relay=1;
  207.                 }                                
  208.                 else                                                          //自动模式
  209.                 {
  210.                         display_string(0x40+5,"AT");
  211.                         if(i<(Hum_L+Hum_H)/2)                   //最大和最小值的平均值作为抽水的判断条件
  212.                         {
  213.                                  relay=0;
  214.                         }
  215.                         else relay=1;
  216.                 }
  217.      }        
  218. }
  219. /********************************************/

  220. /***********************外部中断*********************/
  221. void int0() interrupt 0
  222. {
  223.         delay_50us(20);
  224.         if(key2==0)
  225.         while(!key2);
  226.         flag_motor=~flag_motor;
  227.         if(flag&&flag_jia)
  228.           {
  229.                     //write_com(0x80+9);
  230.               //write_com(0x0f);
  231.                   //delay_50us(1000);
  232.               //write_com(0x0c);
  233.                   //delay_50us(1000);
  234.                   Hum_H++;
  235.                   if(Hum_H>=99) Hum_H=99;
  236.           }
  237.         if(flag&&flag_jian)
  238.           {
  239.                     //write_com(0x80+0x40+9);
  240.                   //write_com(0x0f);
  241.                   //delay_50us(1000);
  242.                   //write_com(0x0c);
  243.                   //delay_50us(1000);
  244.                   Hum_L++;
  245.                   if(Hum_L>=Hum_H) Hum_L=Hum_H-1;
  246.           }   
  247. }

  248. void int1() interrupt 2
  249. {
  250.         delay_50us(20);
  251.         if(key3==0)
  252.         while(!key3);        
  253.         if(flag&&flag_jian)
  254.           {
  255.                     //write_com(0x80+0x40+9);
  256.                   //write_com(0x0f);
  257.                   //delay_50us(1000);
  258.                   //write_com(0x0c);
  259.                   //delay_50us(1000);
  260.                   Hum_L--;
  261.                   if(Hum_L<=1) Hum_L=1;
  262.           }
  263.         if(flag&&flag_jia)
  264.           {
  265.               //write_com(0x80+9);
  266.               //write_com(0x0f);
  267.                   //delay_50us(1000);
  268.               //write_com(0x0c);
  269.                   //delay_50us(1000);
  270.                   Hum_H--;
  271.                   if(Hum_H<=Hum_L) Hum_H=Hum_L+1;
  272.           }
  273. }
  274. /********************************************/
复制代码
全部资料51hei下载地址:
基于51单片机的自动浇花系统.zip (33.96 KB, 下载次数: 991)

评分

参与人数 4黑币 +112 收起 理由
17802971359 + 2 赞一个!
HanCock + 5 共享资料的黑币奖励!
粟米sss + 5 很给力!
admin + 100 共享资料的黑币奖励!

查看全部评分

回复

使用道具 举报

yilinghai 发表于 2018-4-3 10:59 | 显示全部楼层
可以用keil 4编译成功,仿真可以
搜狗截图20180403105826.jpg
回复

使用道具 举报

yxlclp17 发表于 2017-12-5 10:11 | 显示全部楼层
在吗,怎么在keil3_full编译不行的,我一摸一样复制的
回复

使用道具 举报

GeminiYJL 发表于 2017-12-5 12:33 | 显示全部楼层
很厉害的
回复

使用道具 举报

 楼主| dai@1228 发表于 2017-12-8 13:33 | 显示全部楼层
yxlclp17 发表于 2017-12-5 10:11
在吗,怎么在keil3_full编译不行的,我一摸一样复制的

提示什么错误 我用的keil4编译环境
回复

使用道具 举报

y382246268 发表于 2017-12-8 14:30 | 显示全部楼层
赞一个!

评分

参与人数 1黑币 +5 收起 理由
嘿嘿额黑 + 5 共享资料的黑币奖励!

查看全部评分

回复

使用道具 举报

残剑飞雪 发表于 2017-12-9 19:49 | 显示全部楼层
adc0832不能仿真怎么办
回复

使用道具 举报

what-小白 发表于 2018-1-2 12:00 | 显示全部楼层
我是小白,请问左下角的那个圆形在ISIS中怎么找啊
回复

使用道具 举报

1126469277 发表于 2018-2-8 11:49 | 显示全部楼层
yxlclp17 发表于 2017-12-5 10:11
在吗,怎么在keil3_full编译不行的,我一摸一样复制的

格式不对或者缺少什么语句,仔细检查,如果代码没问题建议换keil 4
回复

使用道具 举报

xdl932365660 发表于 2018-2-8 13:25 | 显示全部楼层
给你点赞,暂时不下载 ,需要的时候自然下载
回复

使用道具 举报

大力做大力愛 发表于 2018-2-8 18:45 | 显示全部楼层
没有下载电路吧
回复

使用道具 举报

41797511 发表于 2018-4-1 11:42 | 显示全部楼层
非常给力!!!
回复

使用道具 举报

BOOKS 发表于 2018-4-2 19:09 | 显示全部楼层
很不错,能仿真出来
回复

使用道具 举报

无时崩溃 发表于 2018-4-3 08:05 来自手机 | 显示全部楼层
感谢楼主分享
回复

使用道具 举报

slt201199 发表于 2018-4-15 10:36 | 显示全部楼层
又没有做出来东西
回复

使用道具 举报

icgsgf 发表于 2018-4-15 14:46 来自手机 | 显示全部楼层
身为一个小白感觉很厉害厉害
回复

使用道具 举报

timepasserby 发表于 2018-4-25 17:00 | 显示全部楼层
请问能解释一下AD原理图继电器模块与电源接口模块的工作原理吗,看不太懂。
回复

使用道具 举报

畴昔至稚 发表于 2018-5-5 22:54 | 显示全部楼层
为啥我下载了,却没有清单
回复

使用道具 举报

epaysh 发表于 2018-5-6 05:45 来自手机 | 显示全部楼层
BOOKS 发表于 2018-4-2 19:09
很不错,能仿真出来

你好,我是新手,请问用的是什么仿真软件?能不能发给我一份?545535222@qq.com 谢谢
回复

使用道具 举报

epaysh 发表于 2018-5-6 05:46 来自手机 | 显示全部楼层
你好,我是新手,请问用的是什么仿真软件?能不能发给我一份?545535222@qq.com 谢谢
回复

使用道具 举报

秋天的童话 发表于 2018-5-6 14:35 | 显示全部楼层
厉害了
回复

使用道具 举报

清如许 发表于 2018-5-10 15:46 | 显示全部楼层
很厉害,正好用到
回复

使用道具 举报

17802971359 发表于 2018-5-13 16:30 | 显示全部楼层
绝对干货,不错
回复

使用道具 举报

刘乐涛 发表于 2018-5-15 16:01 | 显示全部楼层
仿真软件叫什么名字   在哪里可以下载,谢谢
回复

使用道具 举报

PaJamas_X 发表于 2018-5-16 16:08 | 显示全部楼层
这个。。。楼主用的是什么湿度传感器呢没找到
回复

使用道具 举报

PaJamas_X 发表于 2018-5-16 16:09 | 显示全部楼层
刘乐涛 发表于 2018-5-15 16:01
仿真软件叫什么名字   在哪里可以下载,谢谢

Proteus
回复

使用道具 举报

li15616313970 发表于 2018-5-20 15:19 | 显示全部楼层
怎么显示屏没有显示啊
回复

使用道具 举报

xianfajushi 发表于 2018-5-23 18:05 | 显示全部楼层
下载一个试看,谢谢分享。
回复

使用道具 举报

echoma 发表于 2018-5-23 20:10 | 显示全部楼层
请问现在很多土壤湿度传感器不需要AD,那程序应该变成什么样子呢?
回复

使用道具 举报

刘乐涛 发表于 2018-5-30 20:24 | 显示全部楼层
各位大哥们,程序有点看不懂,大神能加我好友指点一下吗,1224799169重谢
回复

使用道具 举报

amz 发表于 2018-6-5 19:36 | 显示全部楼层
感觉好厉害呀  新人勿怪
回复

使用道具 举报

hjj12345 发表于 2018-6-25 10:15 | 显示全部楼层
yilinghai 发表于 2018-4-3 10:59
可以用keil 4编译成功,仿真可以

朋友如何仿真啊
回复

使用道具 举报

丑东西 发表于 2018-6-26 18:42 | 显示全部楼层
您好,如果有STC89C51单片机开发板,这个仿真图该怎样设计
回复

使用道具 举报

vbnm 发表于 2018-9-26 22:00 | 显示全部楼层
楼主你好.怎解湿度低于下限值时电机不工作啊
回复

使用道具 举报

cwzhang_87 发表于 2018-10-6 15:14 | 显示全部楼层
多谢分享哈
回复

使用道具 举报

Moqi-S 发表于 2018-11-7 21:24 | 显示全部楼层
感谢楼主分享
回复

使用道具 举报

duyi324 发表于 2018-11-8 09:05 | 显示全部楼层
很给力,谢谢分享
回复

使用道具 举报

wangchang 发表于 2018-11-8 10:29 | 显示全部楼层
谢谢楼主 感谢这次带来的帮助
回复

使用道具 举报

mumu168 发表于 2018-11-9 07:58 | 显示全部楼层
下载学习一下,谢谢分享
回复

使用道具 举报

回忆味 发表于 2018-11-27 08:59 | 显示全部楼层
刚好时实训作业要做赞一个
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表