单片机+MPX4115压力测量仪的Proteus仿真设计+代码+文档

压力测量仪的设计与仿真

一、设计要求:

本课程设计用MPX4115传感器来检测压力参数,ADC0832进行模数转换后,利用单片机AT89C52进行数据处理后,用四个八段数码管显示压力值。压力测量的量程在15.3KPA~114.9KPA,分度值位0.1kpa。

二、构思

2.1 主控模块的选型。

方案一:

采用MSP430系列单片机,该单片机是TI公司1996年开始推向市场的一种16位超低功耗的混合信号处理器。其内部集成了很多模拟电路、数字电路和微处理器,提供强大的功能。不过该芯片昂贵不适合一般的设计开发[3]。

方案二

采用51系列的单片机,该单片机是一个高可靠性,超低价,无法解密,高性能的8位单片机,32个IO口,且STC系列的单片机可以在线编程、调试,方便地实现程序的下载与整机的调试。

因此选用方案二中的51系列单片机作为主控芯片

2.2 显示模块的选型

方案一:

采用点阵式数码管显示,点阵式数码管是由八行八列的发光二极管组成,对于显示文字比较合适,如采用在显示数字显得太浪费,且价格也相对较高,所以不用此种作为显示。

方案二:

采用LED数码管动态扫描,LED数码管价格虽适中,对于显示数字也最合适,而且采用动态扫描法与单片机连接时,占用单片机口线少。

所以本设计中方案二中LED数码管的作为显示模块

2.3 传感器的选型。

压力传感器对于系统至关重要,需要综合实际的需求和各类压力传感器的性 能参数加以选择。一般要选用有温度补偿作用的压力传感器,因为温度补偿特性 可以克服半导体压力传感器件存在的温度漂移问题。

本设计要实现的数字气压计显示的是绝对气压值,同时为了简化电路,提高 稳定性和抗干扰能力,要求使用具有温度补偿能力的压力传感器。MPX4115可以产生高精度模 拟输出电压,数据采集模块由压力传感器MPX4115构成。

三、设计:

3.1 总体设计方案

本次设计是基于8051单片机的测量与显示。利用传感器将压力转换为电信号后,在经过ADC0832的模数转换器经A/D转换后由单片机进行有效处理,然后用数码管进行显示。
 

3.2 单片机最小系统

由AT89C52单片机、时钟电路、复位 电路组成AT89C51是一种带4K字节闪烁可编程可擦除只读存储器的低电压,高性能CMOS8位微处理器,俗称单片机。AT89C52是一种带2K字节闪烁可编程可擦除只读存储器的单片机。单片机的可擦除只读存储器可以反复擦除100次。该器件采用ATMEL高密度非易失存储器制造技术制造,与工业标准的于将多功能MCS-518指令集和输出管脚相兼容。位CPU和闪烁存储器组合在由单个芯片中,ATMEL 的AT89C51是一种高效微控制器,AT89C52是它的一种精简版本。AT89C单片机为很多嵌入式控制系统提供了一种灵活性高且价廉的方案

单片机系统电路

3.3 A/D转换模块

ADC0832是美国国家半导体公司生产的一种8位分辨率、双通道A/D转换芯片。由于它体积小,兼容性,性价比高而深受单片机爱好者及企业欢迎,其目前已经有很高的普及率。学习并使用 ADC0832 可是使我们了解A/D转换器的原理,有助于我们单片机技术水平的提高。8位分辨率双通道A/D转换输入输出电平与TTL/CMOS 相兼容5V电源供电时输入电压在0~5V之间工作频率为250KHZ,转换时间为32μS。 ADC0832为8位分辨率A/D转换芯片,其最高分辨可达256级,可以适应一般的模拟量转换要求。其内部电源输入与参考电压的复用,使得芯片的模拟电压输入在 0~5V 之间。

ADC0832模数转换电路

3.5 传感器模块

MPX4115系列压电电阻传感器是一个硅压力传感器。这个传感器结合了高级微电机技术,薄膜镀金属。还能为高水准模拟输出信号提供一个均衡压力。在0℃-85℃的温度下误差不超过1.5%,温度补偿是-40℃-125℃。为了克服干扰,在MPX4115输出电路中增加了RC滤波电路。如下图所示:

MPX4115信号处理电路

3.6系统总体电路图

四、仿真

通过仿真,调节传感器的+和-,可以观察到数码管数值的变化,压力测量的范围在15.3KPA~114.9KPA之间,基本可以达到要求。

五、心得体会:

传感器课程设计结束了,我的收获很多,我做的是基于AT89C52单片机的压力测量仪的设计,用到的主要芯片是 AT89C52和ADC0808,实现的功能是将传感器采集到的模拟信号转换成单片机可以识别的数字信号,再经单片机转换成数码管可以识别的信息,最后显示输出。通过这些我的硬件和软件开发能力都获得了提高。首先硬件方面,基本了解了电子产品的开发流程和所要做的工作。基本掌握了Proteus原理图的方法,从网上搜索的资料对我帮助很大,遇到不懂的问题自己解决不了,就和大家一起讨论。通过发现问题、分析问题、解决问题,我又学到了许多新的知识,在工程实际应用才能有切身的体会,实践出真知,自己亲自动手去做,才知道知识的匮乏。

单片机源码:

//线性区间标度变换公式:    y=(115-15)/(243-13)*X+15kpa   
#include <AT89X52.h>
#include <intrins.h>
#include <stdio.h>
#define R24C04ADD 0xA1
#define W24C04ADD 0xA0
//ADC0832的引脚
sbit ADCS =P2^2;  //ADC0832 chip seclect
sbit ADDI =P2^4;  //ADC0832 k in
sbit ADDO =P2^4;  //ADC0832 k out
sbit ADCLK =P2^3;  //ADC0832 clock signal
sbit SDA = P2 ^ 1;                                //数据线
sbit SCL = P2 ^ 0;                                //时钟线
bit bAck;                                          //应答标志 当bbAck=1是为正确的应答
unsigned char dispbitcode[8]={0xf7,0xfb,0xfd,0xfe,0xef,0xdf,0xbf,0x7f};  //位扫描
unsigned char dispcode[11]={0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82,0xF8,0x80,0x90,0xff};  //共阳数码管字段码
unsigned char dispbuf[4];
unsigned int temp;
unsigned char getdata; //获取ADC转换回来的值
void delay_1ms(void)  //12mhz delay 1.01ms
{
   unsigned char x,y;   
   x=3;
   while(x--)
  {
       y=40;
       while(y--);
    }
}
void display(void)  //数码管显示函数
{
  char k;
  for(k=0;k<4;k++)
  {
  P1 = dispbitcode[k];
  P0 = dispcode[dispbuf[k]];
  if(k==1)          //加上数码管的dp小数点
          P0&=0x7f;
  delay_1ms();         
  }
}
/************
读ADC0832函数
************/
//采集并返回
unsigned int Adc0832(unsigned char channel)     //AD转换,返回结果
{
    unsigned char i=0;
    unsigned char j;
    unsigned int dat=0;
    unsigned char ndat=0;
    if(channel==0)channel=2;
    if(channel==1)channel=3;
    ADDI=1;
    _nop_();
    _nop_();
    ADCS=0;//拉低CS端
    _nop_();
    _nop_();
    ADCLK=1;//拉高CLK端
    _nop_();
    _nop_();
    ADCLK=0;//拉低CLK端,形成下降沿1
    _nop_();
    _nop_();
    ADCLK=1;//拉高CLK端
    ADDI=channel&0x1;
    _nop_();
    _nop_();
    ADCLK=0;//拉低CLK端,形成下降沿2
    _nop_();
    _nop_();
    ADCLK=1;//拉高CLK端
    ADDI=(channel>>1)&0x1;
    _nop_();
    _nop_();
    ADCLK=0;//拉低CLK端,形成下降沿3
    ADDI=1;//控制命令结束
    _nop_();
    _nop_();
    dat=0;
    for(i=0;i<8;i++)
    {
        dat|=ADDO;//收数据
        ADCLK=1;
        _nop_();
        _nop_();
        ADCLK=0;//形成一次时钟脉冲
        _nop_();
        _nop_();
        dat<<=1;
        if(i==7)dat|=ADDO;
    }  
    for(i=0;i<8;i++)
    {
        j=0;
        j=j|ADDO;//收数据
        ADCLK=1;
        _nop_();
        _nop_();
        ADCLK=0;//形成一次时钟脉冲
        _nop_();
        _nop_();
        j=j<<7;
        ndat=ndat|j;
        if(i<7)ndat>>=1;
    }
    ADCS=1;//拉低CS端
    ADCLK=0;//拉低CLK端
    ADDO=1;//拉高数据端,回到初始状态
    dat<<=8;
    dat|=ndat;
    return(dat);            //return ad k
}
//启动I2C总线,即发送起始条件
void StartI2C()
{
        SDA = 1;                              //发送起始条件数据信号
        _nop_();
        SCL = 1;
        _nop_();                                    //起始建立时间大于4.7us
        _nop_();
        _nop_();
        _nop_();
        _nop_();
        SDA = 0;                              //发送起始信号
        _nop_();
        _nop_();
        _nop_();
        _nop_();
        _nop_();
        SCL = 0;                                //时钟操作
        _nop_();
        _nop_();
}
//结束I2C总线,即发送I2C结束条件
void StopI2C()
{
        SDA = 0;                                //发送结束条件的数据信号
        _nop_();                                      //发送结束条件的时钟信号
        SCL = 1;                                //结束条件建立时间大于4us
        _nop_();
        _nop_();
        _nop_();
        _nop_();
        _nop_();
        SDA = 1;                                //发送I2C总线结束命令
        _nop_();
        _nop_();
        _nop_();
        _nop_();
        _nop_();        
}
//发送一个字节的数据
void        SendByte(unsigned char c)
{
        unsigned char BitCnt;
        for(BitCnt = 0;BitCnt < 8;BitCnt++)                                  //一个字节
                {
                        if((c << BitCnt)& 0x80) SDA = 1;                   //判断发送位
                        else        SDA = 0;
                        _nop_();
                        SCL = 1;                              //时钟线为高,通知从机开始接收数据
                        _nop_();
                        _nop_();
                        _nop_();
                        _nop_();
                        _nop_();
                        SCL = 0;
                }
        _nop_();
        _nop_();
        SDA = 1;                                                //释放数据线,准备接受应答位
        _nop_();
        _nop_();
        SCL = 1;
        _nop_();
        _nop_();
        _nop_();
        if(SDA == 1) bAck =0;
        else bAck = 1;                                                //判断是否收到应答信号
        SCL = 0;
        _nop_();
        _nop_();
}
//接收一个字节的数据
unsigned char RevByte()
{
        unsigned char retc;
        unsigned char BitCnt;
        retc = 0;
        SDA = 1;
        for(BitCnt=0;BitCnt<8;BitCnt++)
        {
                _nop_();
                SCL = 0;                                            //置时钟线为低,准备接收
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                SCL = 1;                                            //置时钟线为高使得数据有效
                _nop_();
                _nop_();
                retc = retc << 1;                                    //左移补零
                if (SDA == 1)
                retc = retc + 1;                                     //当数据为1则收到的数据+1
                _nop_();
                _nop_();
        }
        SCL = 0;
        _nop_();
        _nop_();
        return(retc);                                   //返回收到的数据
}
unsigned char WIICByte(unsigned char WChipAdd,unsigned char InterAdd,unsigned char WIICData)
{
        StartI2C();                                                                                      //启动总线
        SendByte(WChipAdd);                                                        //发送器件地址以及命令
        if (bAck==1)                                                                                          //收到应答
        {
                SendByte(InterAdd);                                                                //发送内部子地址
                if (bAck ==1)
                {
                        SendByte(WIICData);                                                        //发送数据
                        if(bAck == 1)
                        {
                                StopI2C();                    //停止总线
                                return(0xff);
                        }
                        else
                        {
                                return(0x03);
                        }                        
                }
                else
                {
                        return(0x02);
                }
        }
        return(0x01);
}
//读取指定器件的内部指定地址一个字节数据
unsigned char RIICByte(unsigned char WChipAdd,unsigned char RChipAdd,unsigned char InterDataAdd)
{
        unsigned char TempData;        
        TempData = 0;
        StartI2C();                                                                        //启动
        SendByte(WChipAdd);                                                    //发送器件地址以及读命令
        if (bAck==1)                                                                            //收到应答
        {
                SendByte(InterDataAdd);                                        //发送内部子地址
                if (bAck ==1)
                {
                        StartI2C();
                        SendByte(RChipAdd);        
                        if(bAck == 1)
                        {
                                TempData = RevByte();       //接收数据
                                StopI2C();                  //停止I2C总线
                                return(TempData);           //返回数据
                        }
                        else
                        {
                                return(0x03);
                        }        
                }
                else
                {
                        return(0x02);
                }
        }
        else
        {
                return(0x01);
        }
}
void main(void)
{
  unsigned int OverCounter = 0;
  unsigned char ptemp;
  bit OverFlg = 0;
  unsigned int temp,ppress = 0;
  float  press;        
  while(1)
  {         
                     
          getdata=Adc0832(0);
          if(14<getdata<243)                                       //当压力值介于15kpa到115kpa之间时,遵循线性变换
                 {                           
                  int vary=getdata;                                                //y=(115-15)/(243-13)*X+15kpa                        
                        press=((10.0/23.0)*vary)+9.3;                        //测试时补偿值为9.3                                                                                                         
                        temp=(int)(press*10);                  //放大10倍,便于后面的计算
      if(temp != ppress)
      {
        ppress = temp;
        OverFlg = 1;
      }                                                                                            
                        dispbuf[3]=temp/1000;                                     //取压力值百位
                        dispbuf[2]=(temp%1000)/100;                            //取压力值十位
                        dispbuf[1]=((temp%1000)%100)/10;                    //取压力值个位
                        dispbuf[0]=((temp%1000)%100)%10;                        //取压力值十分位
                        display();
      if (temp > 100)
      {
          if(OverFlg == 1)   
          {
            OverCounter++;
            WIICByte(W24C04ADD,0x01,(OverCounter/0xff));    //低位
            WIICByte(W24C04ADD,0x02,(OverCounter%0xff));    //高位
            OverFlg = 0;    //清除标志
          }
      }
           }            
  }
}
————————————————

                            版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
                        
原文链接:https://blog.csdn.net/runweipa/article/details/139134598

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值