找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 2891|回复: 2
收起左侧

51单片机PT100+AD623+MAX1240热电偶测温程序+Proteus仿真图

[复制链接]
ID:591041 发表于 2022-1-18 18:50 | 显示全部楼层 |阅读模式
功能描述:
1、采用51单片机作为主控单元芯片;
2、采用PT100热电偶检测温度;
3、采用AD623作为信号放大电路;
4、采用MAX1240对信号进行模数转换;
5、采用LCD1602作为显示模块;
6、可通过按键修改ADC运算周期;

仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)
51hei.gif
X@I6HL}SG8T30X29T`@W`BG.png

二、    操作说明
KEY1:进入ADC运算周期设置模式;
KEY2: 退出ADC运算周期设置模式;
KEY3:ADC运算周期时间加;
KEY4:ADC运算周期时间减;

点击器件上的按钮,可以调节参数;
51hei.png

单片机源程序如下:
  1. #include <reg52.h>
  2. #include <string.h>
  3. #include <math.h>
  4. #include "intrins.h"

  5. #define ROW1 0x80
  6. #define ROW2 0xC0
  7. #define Ts_CONST 1                //AD采样时间系数,采样时间为0.1xTs_CONST(s)

  8. #define NOACT 0
  9. #define FC 1
  10. #define Start_AD 2
  11. #define Cal_Temperature 3

  12. #define Kpt 0.02732644         //AD芯片max1240的采样值到温度值的转化系数


  13. void LCD_Init(void);
  14. void ClearDisp(unsigned char Row);
  15. void Display(unsigned char Addr, unsigned char *pstr);
  16. void DecToASC(unsigned int Dec,unsigned char *p, unsigned char n);
  17. void Disp_Initize(void);
  18. void Write_CMD(unsigned char CMD);

  19. unsigned int AD_MAX1240(void);

  20. sbit RS=P2^2;
  21. sbit RW=P2^1;
  22. sbit E=P2^0;

  23. sbit SDA=P1^5;
  24. sbit SCL=P1^6;
  25. sbit CS=P1^7;

  26. sbit Key=P3^5;
  27. sbit up=P3^6;
  28. sbit down=P3^7;



  29. unsigned char DispBuf[16];

  30. unsigned char RunState=NOACT;
  31. unsigned int AD_Reslut,tsflag,ts=1;


  32. float  fz,tmp,tmp1,Pt;

  33. void delay(unsigned int j)//延时jms
  34. {
  35.     unsigned int m,n;
  36.     for(n=0;n<j;n++)
  37.       {
  38.        for(m=0;m<120;m++);
  39.        }
  40. }


  41. void main(void)
  42. {        LCD_Init();
  43.         ClearDisp(ROW1);

  44.         Disp_Initize();

  45.         CS=1;
  46.         SDA=1;
  47.         SCL=0;

  48.         TMOD=0x11;
  49.         TF0=0;                //开始程序前先延时65ms,等待max1240上电,以保证AD采样准确
  50.                                 
  51.         TH0=0;
  52.         TL0=0;
  53.         TR0=1;
  54.         while(!TF0);
  55.         TF0=0;
  56.         TR0=0;

  57.         TH0=0x3C;                //12M晶振时延时50ms
  58.         TL0=0xB0;
  59.         ET0=1;
  60.         TR0=1;

  61.         EX1=1;
  62.         IT1=1;

  63.         EA=1;

  64.         while(1)
  65.                 switch(RunState)
  66.                 {        case NOACT:
  67.                                 break;
  68.                         case FC:        //F0计算及显示
  69.                                 fz=pow(10,(tmp-121.0000)/10);
  70.                                 DecToASC(fz,DispBuf,1);
  71.                                 DispBuf[1]='.';
  72.                                 DecToASC((fz-(unsigned int)fz)*1000,DispBuf+2,3);
  73.                                 DispBuf[5]='\0';
  74.                                 Display(0x80+11,DispBuf);
  75.                                        
  76.                                 

  77.                                 RunState=NOACT;
  78.                                 break;
  79.                         case Start_AD:                          //开始AD采样

  80.                                 AD_Reslut=AD_MAX1240();

  81.                                 RunState=Cal_Temperature;
  82.                                 break;
  83.                         case Cal_Temperature:                                     //将AD值转化为温度并进行线性化
  84.                         {
  85.                                 
  86.                                 Pt=AD_Reslut*Kpt;
  87.                                 Pt+=100;          //温度值
  88.                                        

  89.                                 tmp=Pt/100;
  90.                                 tmp=1-tmp;
  91.                                 tmp=2.31e-6L*tmp;
  92.                                 tmp+=1.527480889e-5L;
  93.                                 tmp=sqrt(tmp);
  94.                                 tmp+=-3.9083e-3L;
  95.                                 tmp/=-1.155e-6L;//AD采样值线性化修正
  96.                                 tmp1=tmp;
  97.                                 tmp=(tmp+tmp1)/2;//取平均温度
  98.                                 tmp=((float)((unsigned int)(tmp*10)))/10 ;

  99.                                 DecToASC(tmp,DispBuf,3);
  100.                                 DispBuf[3]='.';
  101.                                 DecToASC((tmp-(unsigned int)tmp)*10,DispBuf+4,1);
  102.                                 DispBuf[5]='\0';
  103.                                 Display(0xC0+2,DispBuf); //显示函数
  104.                           }
  105.         
  106.                                 RunState=NOACT;
  107.                                 break;
  108.                 }
  109. }
  110. /*******************************************************************************/
  111. unsigned int AD_MAX1240(void)        //读取AD,AD芯片为串口数据形式
  112. {        unsigned int adtmp=0;        
  113.         unsigned char i;

  114.         CS=0;
  115.         while(!SDA);

  116.         SCL=1;
  117.         adtmp<<=1;
  118.         SCL=0;

  119.         for(i=0;i<12;i++)
  120.         {        SCL=1;
  121.                 adtmp<<=1;
  122.                 if(SDA==1)adtmp++;
  123.                 SCL=0;
  124.         }

  125.         CS=1;

  126.         return adtmp;
  127. }
  128. /*******************************************************************************/
  129. void Disp_Initize(void)                                //LCD初始化,将Ts,F0,t等提示字符显示在屏幕上
  130. {        //unsigned char *p;

  131.         
  132.         
  133.         DispBuf[0]='T';        
  134.         DispBuf[1]='s';
  135.         DispBuf[2]='=';
  136.         DecToASC(ts,DispBuf+3,2);
  137.         DispBuf[5]='s';
  138.         
  139.         DispBuf[8]='f' ;
  140.         DispBuf[9]='0'        ;
  141.         DispBuf[10]='='        ;
  142.         DispBuf[16]='\0';
  143.         Display(0x80,DispBuf);
  144.         DispBuf[0]='t';
  145.         DispBuf[1]='=';
  146.         DispBuf[2]='\0';
  147.         Display(0xc0,DispBuf);
  148. }
  149. /*******************************************************************************/
  150. void INT1_ISR(void) interrupt 2                          //切换采样时间的中断函数,用于开关量识别和改变Ts(采样时间)
  151. {        
  152.         EA=0;
  153.         TR0=0;
  154.         ET0=0;
  155.         strcpy(DispBuf,"     ");
  156.         Display(0x80+11,DispBuf);

  157.         while(1)
  158.         {
  159.                
  160.                 if(up==0)
  161.                 {
  162.                         delay(100);
  163.                         if(up==0)
  164.                         ts++;

  165.                 }

  166.                
  167.                 if(down==0)
  168.                
  169.                 {   delay(100);
  170.                         if(down==0)
  171.                         ts--;
  172.                 }
  173.                
  174.                 if(ts>60)
  175.                 ts=1;                        
  176.                 if(ts<=0)
  177.                 ts=60;
  178.                 DecToASC(ts,DispBuf,2);
  179.                 DispBuf[2]='s';
  180.                 DispBuf[3]='\0';
  181.                 Display(0x80+3,DispBuf);
  182.                 if(Key==0)
  183.                 break;

  184.         }
  185.         EA=1;
  186.         TR0=1;
  187.         ET0=1;
  188.         
  189. }
  190. /*******************************************************************************/
  191. void T0_ISR(void) interrupt 1         //定时中断程序
  192. {        static unsigned char T0_CNT=Ts_CONST;
  193.         //static bit Flag=0;

  194.         TH0=0x3C;
  195.         TL0=0xB0;

  196.         //Flag=!Flag;

  197.         //if(Flag)return;

  198.         T0_CNT--;
  199.         tsflag++;

  200.         if(!T0_CNT)
  201.         {        T0_CNT=Ts_CONST;
  202.                 RunState=Start_AD; //AD采样
  203.         }
  204.         if(tsflag>=ts*20)
  205.         {
  206.                 tsflag=0;
  207.                 RunState =FC;           //Ts到时进行F0值运算


  208.         }
  209. }
  210. /*******************************************************************************/
  211. void DecToASC(unsigned int Dec,unsigned char *p, unsigned char n)        //将Dec变量编程可以在1602上直接显示的ASICII值
  212. {        unsigned char i;

  213.         p+=n;
  214.         p--;

  215.         for(i=0;i<n;i++)
  216.         {        *p=Dec%10+0x30;
  217.                 p--;
  218.                 Dec/=10;
  219.         }
  220. }
  221. /*LCD显示*******************************************************************************/
  222. void Check_Busy(void)
  223. {//        return;
  224.         do
  225.         {        P0=0xFF;
  226.                 E=0;
  227.                 RS=0;
  228.                 RW=1;
  229.                 E=1;

  230.                 _nop_();
  231.         }while(P0&0x80);

  232.         E=0;
  233. }
  234. /*******************************************************************************/
  235. void Write_CMD(unsigned char CMD)
  236. {        
  237.         Check_Busy();
  238.         E=0;
  239.         RS=0;
  240.         RW=0;
  241.         P0=CMD;
  242.         E=1;
  243.         _nop_();
  244. ……………………

  245. …………限于本文篇幅 余下代码请从51黑下载附件…………
复制代码
所有资料51hei附件下载(含仿真代码等):
热电偶测温-仿真设计(51+PT100+MAX1240+1602+KEY4).7z (9.27 MB, 下载次数: 155)

评分

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

查看全部评分

回复

使用道具 举报

ID:262 发表于 2022-2-23 03:39 | 显示全部楼层
好资料,51黑有你更精彩!!!
回复

使用道具 举报

ID:1111326 发表于 2024-2-21 22:53 | 显示全部楼层
请问一下,我直接搭建ad623仿真时,提示我[SPICE] Too many iterations without convergence.
是什么原因
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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