找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 15381|回复: 6
收起左侧

mpx4115压力测量+ADC0832转换+51单片机电子秤源码与仿真

  [复制链接]
ID:290266 发表于 2018-3-25 03:14 | 显示全部楼层 |阅读模式
分享一个51单片机做的mpx4115电子秤,用ADC0832芯片做的ad转换.
电子秤仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)
0.png

mpx4115压力传感器测量部分:
0.png


单片机源程序如下:
  1. /********************************************************
  2.                    压力测试仪
  3. 系统描述;输入  15--115kPA压力信号
  4.          输出  00h--ffh数字信号(adc0832)
  5.          在LCD上显示实际的压力值,如果超限则报警

  6. 线性区间标度变换公式:    y=(115-15)/(243-13)*X+15kpa   

  7. 作者:
  8. 单位:
  9. 日期:2008.3.7

  10. ********************************************************/

  11. #include<reg51.h>
  12. #include<intrins.h>
  13. #include <absacc.h>
  14. #include <math.h>

  15. #define uchar unsigned char
  16. #define uint   unsigned int
  17. #define BUSY  0x80                               //常量定义
  18. #define DATAPORT P0


  19. //ADC0832的引脚
  20. sbit ADCS =P3^5;  //ADC0832 chip seclect
  21. sbit ADDI =P3^7;  //ADC0832 k in
  22. sbit ADDO =P3^7;  //ADC0832 k out
  23. sbit ADCLK =P3^6;  //ADC0832 clock signal

  24. sbit LCM_RS=P2^0;
  25. sbit LCM_RW=P2^1;
  26. sbit LCM_EN=P2^2;

  27. uchar ad_data;                                    //采样值存储
  28. sbit Alarm_led_red =P3^1;                         //超过压力表量程最大值红色led报警定义
  29. sbit Alarm_led_green=P3^2;                        //低于压力表量程最小值绿色led报警定义
  30.                           //adc采样值存储单元
  31. char press_data;                                  //标度变换存储单元
  32. unsigned char ad_alarm;                           //报警值存储单元
  33. unsigned char press_ge=0;                        //显示值百位
  34. unsigned char press_shifen=0;                        //显示值十位
  35. unsigned char press_baifen=0;                         //显示值个位
  36. unsigned char press_qianfen=0;                        //显示值十分位

  37. uchar code str0[]={"Weight:  .   Kg "};
  38. uchar code str1[]={"Make by GUO TAO "};
  39. uchar code str2[]={"Price:          "};
  40. uchar code str3[]={"Total:          "};
  41. void delay(uint);
  42. void lcd_wait(void);
  43. void delay_LCM(uint);                                                                                                     //LCD延时子程序
  44. void initLCM( void);                                                                                                       //LCD初始化子程序
  45. void lcd_wait(void);                                                                                                      //LCD检测忙子程序
  46. void WriteCommandLCM(uchar WCLCM,uchar BusyC);                              //写指令到ICM子函数
  47. void WriteDataLCM(uchar WDLCM);                                             //写数据到LCM子函数
  48. void DisplayOneChar(uchar X,uchar Y,uchar DData);                           //显示指定坐标的一个字符子函数
  49. void DisplayListChar(uchar X,uchar Y,uchar code *DData);                                         //显示指定坐标的一串字符子函数
  50.   void keyscan();                  
  51. void display(void);                                                         //系统显示子函数
  52. uchar Adc0832(unsigned char channel);                                                                                                                        
  53. void alarm(void);
  54. void data_pro(void);


  55. /**********main funcation************/

  56. void main(void)
  57. {
  58.     delay(500);                      //系统延时500ms启动
  59. //        ad_data=0;                       //采样值存储单元初始化为0
  60.            initLCM( );
  61.         
  62.            WriteCommandLCM(0x01,1);                    //清显示屏
  63.     DisplayListChar(0,0,str0);
  64.            DisplayListChar(0,1,str2);

  65.         while(1)
  66.     {
  67.            ad_data =Adc0832(0);           //采样值存储单元初始化为0
  68.            
  69.             alarm();
  70.         
  71.                 data_pro();
  72.                
  73.                 display();
  74.                                  
  75.         }
  76. }


  77. /*********延时K*1ms,12.000mhz**********/

  78. void delay(uint k)
  79. {
  80.     uint i,j;
  81.     for(i=0;i<k;i++)
  82.     {
  83.         for(j=0;j<60;j++)
  84.                 {;}
  85.     }
  86. }   
  87. /**********写指令到ICM子函数************/

  88. void WriteCommandLCM(uchar WCLCM,uchar BusyC)
  89. {
  90.     if(BusyC)lcd_wait();
  91.         DATAPORT=WCLCM;
  92.     LCM_RS=0;                   // 选中指令寄存器
  93.     LCM_RW=0;                       // 写模式        
  94.     LCM_EN=1;   
  95.         _nop_();   
  96.         _nop_();
  97.         _nop_();
  98.     LCM_EN=0;
  99.    
  100. }

  101. /**********写数据到LCM子函数************/

  102. void WriteDataLCM(uchar WDLCM)
  103. {
  104.     lcd_wait( );             //检测忙信号   
  105.         DATAPORT=WDLCM;
  106.     LCM_RS=1;                // 选中数据寄存器
  107.     LCM_RW=0;                    // 写模式
  108.     LCM_EN=1;
  109.     _nop_();
  110.         _nop_();
  111.         _nop_();
  112.     LCM_EN=0;
  113. }

  114. /***********lcm内部等待函数*************/

  115. void lcd_wait(void)
  116. {
  117.     DATAPORT=0xff;             //读LCD前若单片机输出低电平,而读出LCD为高电平,则冲突,Proteus仿真会有显示逻辑黄色
  118.         LCM_EN=1;
  119.     LCM_RS=0;   
  120.     LCM_RW=1;   
  121.     _nop_();
  122.     _nop_();
  123.         _nop_();
  124.     while(DATAPORT&BUSY)  
  125.         {  LCM_EN=0;
  126.            _nop_();
  127.            _nop_();
  128.            LCM_EN=1;
  129.            _nop_();
  130.            _nop_();
  131.            }         
  132.            LCM_EN=0;        
  133.         
  134. }

  135. /**********LCM初始化子函数***********/

  136. void initLCM( )
  137. {   
  138.         DATAPORT=0;        
  139.         delay(15);
  140.         WriteCommandLCM(0x38,0);    //三次显示模式设置,不检测忙信号
  141.     delay(5);
  142.     WriteCommandLCM(0x38,0);
  143.     delay(5);
  144.     WriteCommandLCM(0x38,0);
  145.     delay(5);

  146.     WriteCommandLCM(0x38,1);    //8bit数据传送,2行显示,5*7字型,检测忙信号
  147.     WriteCommandLCM(0x08,1);    //关闭显示,检测忙信号
  148.     WriteCommandLCM(0x01,1);    //清屏,检测忙信号
  149.     WriteCommandLCM(0x06,1);    //显示光标右移设置,检测忙信号
  150.     WriteCommandLCM(0x0c,1);    //显示屏打开,光标不显示,不闪烁,检测忙信号
  151. }

  152. /****显示指定坐标的一个字符子函数****/

  153. void DisplayOneChar(uchar X,uchar Y,uchar DData)

  154. {
  155.    Y&=0x01;
  156.     X&=0x0f;
  157.     if(Y)X|=0x40;               //若y为1(显示第二行),地址码+0X40
  158.     X|=0x80;                    //指令码为地址码+0X80
  159.     WriteCommandLCM(X,1);
  160.     WriteDataLCM(DData);
  161. }

  162. /*******显示指定坐标的一串字符子函数*****/

  163. void DisplayListChar(uchar X,uchar Y,uchar code *DData)
  164. {
  165.     uchar ListLength=0;
  166.     Y&=0x01;
  167.     X&=0x0f;
  168.     while(X<16)
  169.     {
  170.         DisplayOneChar(X,Y,DData[ListLength]);
  171.         ListLength++;
  172.         X++;
  173.     }
  174. }

  175. /*****************系统显示子函数*****************/

  176. void display(void)
  177. {
  178.            WriteCommandLCM(0x0c,1);                                    //显示屏打开,光标不显示,不闪烁,检测忙信号        
  179.         DisplayListChar(0,0,str0);        
  180.         DisplayListChar(0,1,str2);        


  181.         DisplayOneChar(8,0,press_ge+0x30);
  182.         DisplayOneChar(10,0,press_shifen+0x30);
  183.         DisplayOneChar(11,0,press_baifen+0x30);
  184.     DisplayOneChar(12,0,press_qianfen+0x30);
  185.         delay(1000);                               //稳定显示
  186. }
  187. /************
  188. 读ADC0832函数
  189. ************/

  190. //采集并返回
  191. uchar Adc0832(unsigned char channel)     //AD转换,返回结果
  192. {
  193.     uchar i=0;
  194.     uchar j;
  195.     uint dat=0;
  196.     uchar ndat=0;

  197.     if(channel==0)channel=2;
  198.     if(channel==1)channel=3;
  199.     ADDI=1;
  200.     _nop_();
  201.     _nop_();
  202.     ADCS=0;//拉低CS端
  203.     _nop_();
  204.     _nop_();
  205.     ADCLK=1;//拉高CLK端
  206.     _nop_();
  207.     _nop_();
  208.     ADCLK=0;//拉低CLK端,形成下降沿1
  209.     _nop_();
  210.     _nop_();
  211.     ADCLK=1;//拉高CLK端
  212.     ADDI=channel&0x1;
  213.     _nop_();
  214.     _nop_();
  215.     ADCLK=0;//拉低CLK端,形成下降沿2
  216.     _nop_();
  217.     _nop_();
  218.     ADCLK=1;//拉高CLK端
  219.     ADDI=(channel>>1)&0x1;
  220.     _nop_();
  221.     _nop_();
  222.     ADCLK=0;//拉低CLK端,形成下降沿3
  223.     ADDI=1;//控制命令结束
  224.     _nop_();
  225.     _nop_();
  226.     dat=0;
  227.     for(i=0;i<8;i++)
  228.     {
  229.         dat|=ADDO;//收数据
  230.         ADCLK=1;
  231.         _nop_();
  232.         _nop_();
  233.         ADCLK=0;//形成一次时钟脉冲
  234.         _nop_();
  235.         _nop_();
  236.         dat<<=1;
  237.         if(i==7)dat|=ADDO;
  238.     }  
  239.     for(i=0;i<8;i++)
  240.     {
  241.         j=0;
  242.         j=j|ADDO;//收数据
  243.         ADCLK=1;
  244.         _nop_();
  245.         _nop_();
  246.         ADCLK=0;//形成一次时钟脉冲
  247.         _nop_();
  248.         _nop_();
  249.         j=j<<7;
  250.         ndat=ndat|j;
  251.         if(i<7)ndat>>=1;
  252.     }
  253.     ADCS=1;//拉低CS端
  254.     ADCLK=0;//拉低CLK端
  255.     ADDO=1;//拉高数据端,回到初始状态
  256.     dat<<=8;
  257.     dat|=ndat;
  258.     return(dat);            //return ad k
  259. }


  260. void data_pro(void)
  261. {
  262.       unsigned int temp;
  263.       float  press;                             

  264.           if(14<ad_data<243)                                       //当压力值介于15kpa到115kpa之间时,遵循线性变换
  265.                  {                           
  266.                     int vary=ad_data;                                                //y=(115-15)/(243-13)*X+15kpa                        
  267.                         press=((10.0/23.0)*vary)+9.3;                        //测试时补偿值为9.3
  268.                                                                                                                   
  269.                         temp=(int)(press*10);                        //放大10倍,便于后面的计算
  270.                                                                                                
  271.                                                                                                             
  272.                         press_ge=temp/1000;                                     //取压力值百位
  273.                         press_shifen=(temp%1000)/100;                            //取压力值十位
  274.                         press_baifen=((temp%1000)%100)/10;                    //取压力值个位
  275.                         press_qianfen=((temp%1000)%100)%10;                        //取压力值十分位

  276.                       }            
  277.                         
  278. }
  279. /*****************报警子函数*******************/

  280. void alarm(void)
  281. {
  282.          if(ad_data>=243)                           //如果当前压力值大于115kpa,

  283.           {  Alarm_led_red=0; }                 //则启动red led报警
  284.         
  285.           else
  286.              {          Alarm_led_red=1; }                        //关闭red led 报警
  287.                   
  288.         
  289.           if(ad_data<=14)                                                        //如果当前压力值小于16kpa

  290.              {  Alarm_led_green=0;  }               //则启动green led报警

  291.       else
  292.              {   Alarm_led_green=1; }                              //关闭green  led 报警
  293.          
  294. }


复制代码
0.jpg
所有资料51hei提供下载:
http://www.51hei.com/bbs/dpj-33999-1.html



回复

使用道具 举报

ID:312493 发表于 2018-4-25 09:37 | 显示全部楼层
感谢分享
回复

使用道具 举报

ID:432801 发表于 2018-11-25 15:48 来自手机 | 显示全部楼层
我想问,程序烧进去怎么看现象的变化total那儿没变化
回复

使用道具 举报

ID:571534 发表于 2019-6-25 15:39 | 显示全部楼层
左下面的逻辑门是什么,看不清
回复

使用道具 举报

ID:777921 发表于 2020-6-27 14:01 | 显示全部楼层
这个传感器模拟电路能不能换成MPX4115传感器呀楼主
回复

使用道具 举报

ID:805563 发表于 2020-7-19 22:24 来自手机 | 显示全部楼层
zhouyidahaoren 发表于 2020-6-27 14:01
这个传感器模拟电路能不能换成MPX4115传感器呀楼主

你下载文章了吗?我觉得可以换的。
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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