找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 7268|回复: 13
收起左侧

STC12C5A60S2单片机PID温度调节程序

  [复制链接]
ID:379018 发表于 2018-12-6 19:27 | 显示全部楼层 |阅读模式
采用DS18B20采集温度,电热丝加热。或者是专用模块。

单片机源程序如下:
  1. #include "pid.h"

  2. PID pid; //存放PID算法所需要的数据


  3. void PID_Calc()  //pid计算
  4. {
  5.         float DelEk;
  6.         float ti,ki;
  7.         float td;
  8.         float kd;
  9.         float out;

  10. if(pid.C1ms<(pid.T/10))  //计算周期未到
  11. {
  12.     return ;
  13. }

  14. pid.Ek=pid.Sv-pid.Pv;   //得到当前的偏差值
  15. pid.Pout=pid.Kp*pid.Ek;      //比例输出

  16. pid.SEk+=pid.Ek;        //历史偏差总和

  17. DelEk=pid.Ek-pid.Ek_1;  //最近两次偏差之差

  18. ti=pid.T/pid.Ti;
  19. ki=ti*pid.Kp;

  20.   pid.Iout=ki*pid.SEk;//*pid.Kp;  //积分输出

  21. td=pid.Td/pid.T;
  22. kd=pid.Kp*td;

  23.   pid.Dout=kd*DelEk;    //微分输出

  24. out= pid.Pout+ pid.Iout+ pid.Dout;

  25. //////////////////////////////////////////////////////////

  26. if(out>pid.pwmcycle)
  27. {
  28.   pid.OUT=pid.pwmcycle;
  29. }
  30. else if(out<0)
  31. {
  32.   pid.OUT=pid.OUT0;
  33. }
  34. else
  35. {
  36.   pid.OUT=out;
  37. }
  38. pid.Ek_1=pid.Ek;  //更新偏差

  39. pid.C1ms=0;

  40. }

  41. /*====================================
  42. 初始化  PID  函数
  43. =================================*/
  44. void PID_Init()
  45. {
  46.     pid.Sv=40;//用户设定温度
  47.         pid.Kp=30;
  48.         pid.T=1000;//PID计算周期
  49.     pid.Ti=500000;//积分时间
  50.         pid.Td=1000;//微分时间
  51.         pid.pwmcycle=200;//pwm周期1000
  52.         pid.OUT0=10;
  53. }
复制代码

主程序:
  1. #include "DS18B20.h"
  2. #include "pid.h"
  3. #include "LCD.H"

  4. sbit pwmout=P1^5;          //PWM输出口,此口可任意设定
  5. sbit wela=P2^7;
  6. sbit dula=P2^6;
  7. uchar count;           //占空比最小单位,也是以后的计数值
  8. void PID_out()  //输出PID运算结果到负载---每1ms被调用1次
  9. {
  10.    static u16 pw;
  11.          pw++;
  12.          if(pw>=pid.pwmcycle)  //
  13.          {
  14.            pw=0;
  15.          }
  16.           //0  ~  pid.pwmcycle-1
  17.          
  18.          if(pw<pid.OUT)
  19.          {
  20.            pwmout=0;//加热
  21.          }
  22.          else
  23.          {
  24.            pwmout=1;//停止加热
  25.          }
  26.          
  27. }
  28. void init()
  29. {
  30.         TMOD=0x11;          
  31.         TH1=(65536-9174)/256;           //65526/256         这两句记录1个脉冲,即10毫秒产生一个中断
  32.         TL1=(65536-9174)%256;           //65526%256         
  33.         TH0=(65536-9174)/256;           //65526/256         这两句记录500个脉冲,即10m秒产生一个中断
  34.         TL0=(65536-9174)%256;           //65526%256         
  35.         EA=1;                   //开总中断
  36.         ET0=1;                   //开定时器0中断
  37.         ET1=1;                   //开定时器1中断
  38.         pwmout=0;                   //先置输出口为低电平
  39.         TR0=1;                        //以上设置完毕后,最后开启定时器0
  40.         TR1=1;                        //以上设置完毕后,最后开启定时器1
  41. }
  42. void tim0() interrupt 1                 //定时器0中断函数
  43. {

  44.         TH0=(65536-9174)/256;
  45.         TL0=(65536-9174)%256;
  46.         pid.C1ms++;
  47.         PID_Calc();       
  48.        
  49. }
  50. /**/         
  51. void tim1() interrupt 3                 //定时器0中断函数
  52. {
  53.         TH1=(65536-9174)/256;
  54.         TL1=(65536-9174)%256;                 
  55.         PID_out();
  56. }

  57. void main()
  58. {
  59.         uint i;
  60.         uchar datas1[]={0,0,0};
  61.         wela=0;
  62.         dula=0;
  63.         delay(2);
  64.         lcd_init();       
  65.         PID_Init();       
  66.         init();
  67. /*        ds_init( );//初始化DS18B20
  68.         write_byte(0xcc);//发送跳跃ROM指令
  69.         write_byte(0x4e);//写暂存器指令
  70.         write_byte(0x7f);
  71.         write_byte(0xf7);
  72.         write_byte(0x1f);//配置工作在9位模式下
  73.         ds_init();//初始化DS18B20
  74.         write_byte(0xcc);//发送跳跃ROM指令
  75.         write_byte(0x48);         */       
  76.                 while(1)
  77.                 {                  
  78.                 datas1[2]=(int)pid.OUT%1000/100;
  79.                 datas1[3]=(int)pid.OUT%100/10;
  80.                 datas1[4]=(int)pid.OUT%10;
  81.                 lcd_pos(2,0);
  82.                 for(i=2;i<5;i++)
  83.                 write_dat(0x30+datas1[i]);       
  84.                 if(pid.C1ms>(pid.T/11))
  85.                 {
  86.                 Display_temp();
  87.                 }       
  88.                 }
  89. }
复制代码


所有资料51hei提供下载:
PID温度控制.zip (50.08 KB, 下载次数: 330)
回复

使用道具 举报

ID:277550 发表于 2018-12-31 21:58 | 显示全部楼层
不错,不过少了lcd.c文件
回复

使用道具 举报

ID:412814 发表于 2019-1-25 14:27 | 显示全部楼层
这个真挺好,先研究下。谢谢楼主
回复

使用道具 举报

ID:475247 发表于 2019-2-10 16:54 | 显示全部楼层
devcang 发表于 2018-12-31 21:58
不错,不过少了lcd.c文件

LCD就用不到吧,毕竟是很基础的东西

评分

参与人数 2黑币 +47 收起 理由
春风十里吹 + 12 很给力!
jinsheng7533967 + 35

查看全部评分

回复

使用道具 举报

ID:71618 发表于 2019-4-25 11:09 | 显示全部楼层
学习一下!
回复

使用道具 举报

ID:524445 发表于 2019-4-29 15:11 | 显示全部楼层
不错!比较简单,好理解!
回复

使用道具 举报

ID:607640 发表于 2019-11-4 20:01 | 显示全部楼层
你真是个好家伙,谢谢分享!
回复

使用道具 举报

ID:607640 发表于 2019-11-4 20:02 | 显示全部楼层
感谢分享!
回复

使用道具 举报

ID:97678 发表于 2019-12-4 20:43 | 显示全部楼层
初学者,最好还是文件完整,学习起来比较方便。希望补充完整
回复

使用道具 举报

ID:676336 发表于 2020-2-20 20:27 | 显示全部楼层
为什么if(pw<pid.out)就加热,不然就停止加热,这里不懂
回复

使用道具 举报

ID:299193 发表于 2020-3-11 01:03 | 显示全部楼层
真不错,谢谢了!
回复

使用道具 举报

ID:299193 发表于 2020-3-13 22:41 | 显示全部楼层
不错,好好学习一下
回复

使用道具 举报

ID:364328 发表于 2020-3-19 21:10 | 显示全部楼层
只有程序没有电路图吗请问
回复

使用道具 举报

ID:239055 发表于 2020-3-20 07:07 来自手机 | 显示全部楼层
学习了,51黑的资源有点少,希望大家多参与
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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