找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 3919|回复: 4
收起左侧

51单片机超声波液位仪Proteus仿真原理图程序PCB文件 带温度补偿

[复制链接]
ID:604514 发表于 2021-8-3 17:31 | 显示全部楼层 |阅读模式
全套51单片机超声波液位仪资料,含程序、仿真、PCB文件,做出来的实物能实现功能。利用超声波测液位,用LCD1602现实,带温度补偿,带报警指示。PCB蜂鸣器处忘加驱动,电流不够,声音不够大,可换成灯。
初始零液位距离为15CM,报警液位为低于5CM,低于5CM时大灯点亮。
如需要校准改变零液位:
点击设置,设置指示灯亮起
点击液位归零
再次点击设置,指示灯熄灭,开始正常工作。
如需改变报警液位:
点击设置
点击加或者减
换位可以改变按加减时改变个位还是十分位的值,默认为十分位。
仿真说明:
仿真需要先校准归零。
如第一次仿真失败,方法如下:
停止仿真;
双击MCU,出现对话框;
点击program file栏目后的文件夹图标,出现文件选择框;
选择chengxu--output--双击HC-SR04.hex
点击确定

仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)
51hei.png

Altium Designer画的原理图和PCB图如下:(51hei附件中可下载工程文件)
51hei.png 51hei.png

单片机源程序如下:
  1. #include "config.h"
  2. #include "Lcd1602.h"
  3. #include "Ultrasonic.h"
  4. #include <reg52.h>

  5. uint8 flag;   //定义定时标志
  6. uint16 set_s;  //设定值
  7. uint16 lv_0;   //零位
  8. uint16 length;
  9. uint16 lv_n;

  10. bit flag1s = 0;          //1s定时标志
  11. bit fusu = 0;          //1s定时标志
  12. bit setflag= 0;          //进入设置模式标志
  13. bit changewei=0;
  14. unsigned char T0RH = 0;  //T0重载值的高字节
  15. unsigned char T0RL = 0;  //T0重载值的低字节
  16. //
  17. void ConfigTimer0(unsigned int ms);
  18. unsigned char IntToString(unsigned char *str, int dat);
  19. extern bit Start18B20();
  20. extern bit Get18B20Temp(int *temp);

  21. sbit led=P2^0;
  22. sbit bz=P1^5;
  23. sbit k3=P1^7;         //更换十位和百位
  24. sbit k1=P3^0;         //加
  25. sbit k2=P3^1;         //减
  26. sbit k4=P1^6;         //设置零位
  27. void ShowDistance();
  28. void ShowSets();
  29. void delayms(unsigned int ms);

  30. /* 主函数 */
  31. void main()
  32. {        
  33.            bit res;
  34.                
  35.     int temp;        //读取到的当前温度值
  36.     int intT, decT;  //温度值的整数和小数部分
  37.     unsigned char len;
  38.     unsigned char str[12];
  39.     set_s=50;//mm
  40.         lv_0=150;//mm
  41.     EA = 1;            //开启总中断
  42.     TMOD &= 0Xf0;
  43.     TMOD |= 0X01;      //设置T0为方式1
  44.     TH0 = 0xFC;        //为T0赋初值0xFC67,定时1ms
  45.     TL0 = 0x67;
  46.     ET0 = 1;           //使能T0中断
  47.     TR0 = 1;           //启动T0
  48.         IT0=1;        //设置外部中断0的触发方式为下降沿
  49.         EX0=1;        //开启外部中断
  50.     InitUltrasonic();  //超声波模块初始化
  51.     InitLcd1602();     //1602初始化
  52.         Start18B20();      //启动DS18B20
  53.         LcdShowStr(0, 0,"SETD:");  //显示距离单位
  54.         LcdShowStr(0, 1,"LV:");  //显示距离单位
  55.         LcdShowStr(10, 1,"T:");  //显示距离单位
  56.         k1=1;

  57.     while(1)
  58.     {
  59.         if(flag == 1)
  60.         {
  61.             flag = 0;
  62.                         ShowSets();
  63.             ShowDistance();
  64.                     if (1)  //每秒更新一次温度
  65.         {
  66.             res = Get18B20Temp(&temp);  //读取当前温度
  67.             if (res)                    //读取成功时,刷新当前温度显示
  68.             {
  69.                 intT = temp >> 4;             //分离出温度值整数部分
  70.                 decT = temp & 0xF;            //分离出温度值小数部分
  71.                 len = IntToString(str, intT); //整数部分转换为字符串
  72.                 str[len++] = '.';             //添加小数点
  73.                 decT = (decT*10) / 16;        //二进制的小数部分转换为1位十进制位
  74.                 str[len++] = decT + '0';      //十进制小数位再转换为ASCII字符
  75.                 while (len < 6)               //用空格补齐到6个字符长度
  76.                 {
  77.                     str[len++] = ' ';
  78.                 }
  79.                 str[len] = '\0';              //添加字符串结束符
  80.                 LcdShowStr(12, 1, str);        //显示到液晶屏上
  81.             }
  82.             else                        //读取失败时,提示错误信息
  83.             {
  84.                 LcdShowStr(12, 1, "error!");
  85.             }
  86.             Start18B20();               //重新启动下一次转换
  87.         }
  88.         }
  89.                 while(setflag){//按下设置键,进入设置模式,循环直到再一次按下设置键

  90.                         if(k1==0)                  //检测按键K1是否按下
  91.                                 {        
  92.                                         delayms(5);   //消除抖动 一般大约10ms
  93.                                         if(k1==0)         //再次判断按键是否按下
  94.                                         {  
  95.                                                 if(changewei==0)set_s++;
  96.                                                 else set_s+=10;
  97.                                         }
  98.                                         while(!k1);         //检测按键是否松开
  99.                                 }
  100.                         if(k2==0)                  //检测按键K1是否按下
  101.                                 {        
  102.                                         delayms(5);   //消除抖动 一般大约10ms
  103.                                         if(k2==0)         //再次判断按键是否按下
  104.                                         {
  105.                                                 if(changewei==0)set_s--;
  106.                                                         else set_s-=10;
  107.                                         }
  108.                                         while(!k2);         //检测按键是否松开
  109.                                 }
  110.                         if(k3==0)                  //检测按键K1是否按下
  111.                                 {        
  112.                                         delayms(5);   //消除抖动 一般大约10ms
  113.                                         if(k3==0)         //再次判断按键是否按下
  114.                                         {
  115.                                                 changewei=~changewei;
  116.                                         }
  117.                                         while(!k3);         //检测按键是否松开
  118.                                 }
  119.                         if(k4==0)                  //检测按键K1是否按下
  120.                                 {        
  121.                                         delayms(5);   //消除抖动 一般大约10ms
  122.                                         if(k4==0)         //再次判断按键是否按下
  123.                                         {
  124.                                          lv_0=length;
  125.                                         }
  126.                                         while(!k4);         //检测按键是否松开
  127.                                 }
  128.                                 ShowSets();
  129.                 }
  130.                 if(lv_n>set_s) bz=1;
  131.                 else bz=0;
  132.     }
  133. }
  134. unsigned char IntToString(unsigned char *str, int dat)
  135. {
  136.     signed char i = 0;
  137.     unsigned char len = 0;
  138.     unsigned char buf[6];
  139.    
  140.     if (dat < 0)  //如果为负数,首先取绝对值,并在指针上添加负号
  141.     {
  142.         dat = -dat;
  143.         *str++ = '-';
  144.         len++;
  145.     }
  146.     do {          //先转换为低位在前的十进制数组
  147.         buf[i++] = dat % 10;
  148.         dat /= 10;
  149.     } while (dat > 0);
  150.     len += i;     //i最后的值就是有效字符的个数
  151.     while (i-- > 0)   //将数组值转换为ASCII码反向拷贝到接收指针上
  152.     {
  153.         *str++ = buf[i] + '0';
  154.     }
  155.     *str = '\0';  //添加字符串结束符
  156.    
  157.     return len;   //返回字符串长度
  158. }




  159. /* 执行测距并显示结果 */
  160. void ShowDistance()
  161. {   
  162.     uint8 i;
  163.     uint8 buff[4];
  164.         uint8 buff2[5];
  165.     uint16 Distance ;   //暂存距离值
  166.     Distance = GetDistance();
  167.         length = Distance;//mm
  168.         if (Distance < lv_0)
  169.                 {
  170.                 Distance = lv_0-Distance;
  171.                 lv_n= Distance;
  172.                 fusu=0;
  173.                 }
  174.         else
  175.          {
  176.                  Distance =Distance- lv_0;
  177.                  lv_n=0;
  178.                  fusu=1;
  179.          }
  180.         if((Distance<999)&&(fusu==0)){
  181.             buff[3] = Distance%10 + '0';   //将数据转换为字符形式
  182.             buff[2] ='.';
  183.             buff[1] = (Distance/10)%10 + '0';
  184.             buff[0] = (Distance/100)%10 + '0';
  185.             for(i = 0; i <= 0; i++)  //整数部分高位的0转换为空白符
  186.             {
  187.                 if(buff[i] == '0')   //判断是不是字符0
  188.                     buff[i] = ' ';   //将字符0 转换为空格 即不显示
  189.                 else
  190.                     break;
  191.             }
  192.             LcdShowStr(3, 1, &buff); //显示距离值
  193.             LcdShowStr(7, 1, "cm");  //显示距离单位
  194.         }
  195.                 if((Distance<99)&&(fusu==1)){
  196.             buff[3] =' ';   //将数据转换为字符形式
  197.             buff[2] =Distance%10 + '0';
  198.             buff[1] ='.';
  199.             buff[0] = (Distance/10)%10 + '0';
  200. //            for(i = 0; i <= 0; i++)  //整数部分高位的0转换为空白符
  201. //            {
  202. //                if(buff[i] == '0')   //判断是不是字符0
  203. //                    buff[i] = ' ';   //将字符0 转换为空格 即不显示
  204. //                else
  205. //                    break;
  206. //            }
  207.                         LcdShowStr(3, 1, "-");  //显示距离单位
  208.             LcdShowStr(4, 1, &buff); //显示距离值
  209.             LcdShowStr(7, 1, "cm");  //显示距离单位
  210.         }
  211.         if(Distance>999||((Distance>99)&&(fusu==1))){
  212.                 LcdShowStr(3, 1, "ERROR");  //显示距离单位
  213. //            else {
  214. //                        buff2[4] = Distance%10 + '0';   //将数据转换为字符形式
  215. //                    buff2[3] ='.';
  216. //                    buff2[2] = (Distance/10)%10 + '0';
  217. //                    buff2[1] = (Distance/100)%10 + '0';
  218. //                        buff2[0] = (Distance/1000)%10 + '0';
  219. //                    for(i = 0; i <= 3; i++)  //整数部分高位的0转换为空白符
  220. //                    {
  221. //                        if(buff2[i] == '0')   //判断是不是字符0
  222. //                            buff2[i] = ' ';   //将字符0 转换为空格 即不显示
  223. //                        else
  224. //                            break;
  225. //                    }
  226. //                    LcdShowStr(3, 1, &buff2); //显示距离值
  227. //                    LcdShowStr(8, 1, "cm");  //显示距离单位
  228.         //        }
  229.         }
  230. }
  231. void ShowSets()
  232. {   
  233.     uint8 i;
  234.     uint8 buff[4];

  235.         if(set_s>999)set_s=999;
  236.     buff[3] = set_s%10 + '0';   //将数据转换为字符形式
  237.     buff[2] ='.';
  238.     buff[1] = (set_s/10)%10 + '0';
  239.     buff[0] = (set_s/100)%10 + '0';
  240.     for(i = 0; i <= 3; i++)  //整数部分高位的0转换为空白符
  241.     {
  242.         if(buff[i] == '0')   //判断是不是字符0
  243.             buff[i] = ' ';   //将字符0 转换为空格 即不显示
  244.         else
  245.             break;
  246.     }
  247.     LcdShowStr(6, 0, &buff); //显示距离值
  248.     LcdShowStr(10, 0, "cm");  //显示距离单位
  249. }

  250. /* 定时器0中断服务函数 */
  251. ……………………

  252. …………限于本文篇幅 余下代码请从51黑下载附件…………
复制代码
51hei.png
所有资料51hei附件下载:
超声波液位仪.7z (131.92 KB, 下载次数: 130)
回复

使用道具 举报

ID:328014 发表于 2021-8-7 16:18 | 显示全部楼层
好资料,51黑有你更精彩!!!
回复

使用道具 举报

ID:66328 发表于 2021-8-9 13:11 | 显示全部楼层
大神牛啊,那个超声波的仿真模块在哪里下的
回复

使用道具 举报

ID:1076417 发表于 2023-5-9 19:30 | 显示全部楼层
hnqylgq 发表于 2021-8-9 13:11
大神牛啊,那个超声波的仿真模块在哪里下的

Proteus8.9里面自带
回复

使用道具 举报

ID:1113285 发表于 2024-3-20 14:50 | 显示全部楼层
好资料,加个电机驱动就完美了
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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