标题:
STC12C5A60S2单片机PID温度调节程序
[打印本页]
作者:
qqyy123232
时间:
2018-12-6 19:27
标题:
STC12C5A60S2单片机PID温度调节程序
采用DS18B20采集温度,电热丝加热。或者是专用模块。
单片机源程序如下:
#include "pid.h"
PID pid; //存放PID算法所需要的数据
void PID_Calc() //pid计算
{
float DelEk;
float ti,ki;
float td;
float kd;
float out;
if(pid.C1ms<(pid.T/10)) //计算周期未到
{
return ;
}
pid.Ek=pid.Sv-pid.Pv; //得到当前的偏差值
pid.Pout=pid.Kp*pid.Ek; //比例输出
pid.SEk+=pid.Ek; //历史偏差总和
DelEk=pid.Ek-pid.Ek_1; //最近两次偏差之差
ti=pid.T/pid.Ti;
ki=ti*pid.Kp;
pid.Iout=ki*pid.SEk;//*pid.Kp; //积分输出
td=pid.Td/pid.T;
kd=pid.Kp*td;
pid.Dout=kd*DelEk; //微分输出
out= pid.Pout+ pid.Iout+ pid.Dout;
//////////////////////////////////////////////////////////
if(out>pid.pwmcycle)
{
pid.OUT=pid.pwmcycle;
}
else if(out<0)
{
pid.OUT=pid.OUT0;
}
else
{
pid.OUT=out;
}
pid.Ek_1=pid.Ek; //更新偏差
pid.C1ms=0;
}
/*====================================
初始化 PID 函数
=================================*/
void PID_Init()
{
pid.Sv=40;//用户设定温度
pid.Kp=30;
pid.T=1000;//PID计算周期
pid.Ti=500000;//积分时间
pid.Td=1000;//微分时间
pid.pwmcycle=200;//pwm周期1000
pid.OUT0=10;
}
复制代码
主程序:
#include "DS18B20.h"
#include "pid.h"
#include "LCD.H"
sbit pwmout=P1^5; //PWM输出口,此口可任意设定
sbit wela=P2^7;
sbit dula=P2^6;
uchar count; //占空比最小单位,也是以后的计数值
void PID_out() //输出PID运算结果到负载---每1ms被调用1次
{
static u16 pw;
pw++;
if(pw>=pid.pwmcycle) //
{
pw=0;
}
//0 ~ pid.pwmcycle-1
if(pw<pid.OUT)
{
pwmout=0;//加热
}
else
{
pwmout=1;//停止加热
}
}
void init()
{
TMOD=0x11;
TH1=(65536-9174)/256; //65526/256 这两句记录1个脉冲,即10毫秒产生一个中断
TL1=(65536-9174)%256; //65526%256
TH0=(65536-9174)/256; //65526/256 这两句记录500个脉冲,即10m秒产生一个中断
TL0=(65536-9174)%256; //65526%256
EA=1; //开总中断
ET0=1; //开定时器0中断
ET1=1; //开定时器1中断
pwmout=0; //先置输出口为低电平
TR0=1; //以上设置完毕后,最后开启定时器0
TR1=1; //以上设置完毕后,最后开启定时器1
}
void tim0() interrupt 1 //定时器0中断函数
{
TH0=(65536-9174)/256;
TL0=(65536-9174)%256;
pid.C1ms++;
PID_Calc();
}
/**/
void tim1() interrupt 3 //定时器0中断函数
{
TH1=(65536-9174)/256;
TL1=(65536-9174)%256;
PID_out();
}
void main()
{
uint i;
uchar datas1[]={0,0,0};
wela=0;
dula=0;
delay(2);
lcd_init();
PID_Init();
init();
/* ds_init( );//初始化DS18B20
write_byte(0xcc);//发送跳跃ROM指令
write_byte(0x4e);//写暂存器指令
write_byte(0x7f);
write_byte(0xf7);
write_byte(0x1f);//配置工作在9位模式下
ds_init();//初始化DS18B20
write_byte(0xcc);//发送跳跃ROM指令
write_byte(0x48); */
while(1)
{
datas1[2]=(int)pid.OUT%1000/100;
datas1[3]=(int)pid.OUT%100/10;
datas1[4]=(int)pid.OUT%10;
lcd_pos(2,0);
for(i=2;i<5;i++)
write_dat(0x30+datas1[i]);
if(pid.C1ms>(pid.T/11))
{
Display_temp();
}
}
}
复制代码
所有资料51hei提供下载:
PID温度控制.zip
(50.08 KB, 下载次数: 335)
2018-12-6 19:26 上传
点击文件名下载附件
下载积分: 黑币 -5
作者:
devcang
时间:
2018-12-31 21:58
不错,不过少了lcd.c文件
作者:
jifengjianwu
时间:
2019-1-25 14:27
这个真挺好,先研究下。谢谢楼主
作者:
还有谁?
时间:
2019-2-10 16:54
devcang 发表于 2018-12-31 21:58
不错,不过少了lcd.c文件
LCD就用不到吧,毕竟是很基础的东西
作者:
xshzwx
时间:
2019-4-25 11:09
学习一下!
作者:
sungreatfriend
时间:
2019-4-29 15:11
不错!比较简单,好理解!
作者:
贾长安
时间:
2019-11-4 20:01
你真是个好家伙,谢谢分享!
作者:
贾长安
时间:
2019-11-4 20:02
感谢分享!
作者:
jovew
时间:
2019-12-4 20:43
初学者,最好还是文件完整,学习起来比较方便。希望补充完整
作者:
了好久那本书
时间:
2020-2-20 20:27
为什么if(pw<pid.out)就加热,不然就停止加热,这里不懂
作者:
zhangweiyu
时间:
2020-3-11 01:03
真不错,谢谢了!
作者:
zhangweiyu
时间:
2020-3-13 22:41
不错,好好学习一下
作者:
qwthh
时间:
2020-3-19 21:10
只有程序没有电路图吗请问
作者:
c20160526
时间:
2020-3-20 07:07
学习了,51黑的资源有点少,希望大家多参与
欢迎光临 (http://www.51hei.com/bbs/)
Powered by Discuz! X3.1