『壹』 溫度控制的PID的c語言程序中怎麼進行參數自整定
溫度控制常用的自整定方法為極限環法,也叫繼電振盪法,就是讓溫度在設定值上下振盪2~3次,之後通過振盪的波形、時間計算出增益和周期!
『貳』 如何用c語言實現PID演算法的參數計算
這個問題屬於PID的自整定,有簡單的繼電器演算法,我試過,效果不理想。
說了半天,我也沒找到很好用的自整定程序,呵呵。
如果你找到好用的,希望能夠分享一下哦。
『叄』 什麼是pid演算法,難學嗎,用C語言,plc怎麼實現
一、什麼是PID:
PID即:Proportional(比例)、Integral(積分)、Differential(微分)的縮寫。顧名思義,PID控制演算法是結合比例、積分和微分三種環節於一體的控制演算法,它是連續系統中技術最為成熟、應用最為廣泛的一種控制演算法,該控制演算法出現於20世紀30至40年代,適用於對被控對象模型了解不清楚的場合。 ---網路
二、PID是否難學:
在工業應用中PID及其衍生演算法是應用最廣泛的演算法之一,是當之無愧的萬能演算法,如果能夠熟練掌握PID演算法的設計與實現過程,對於一般的研發人員來講,應該是足夠應對一般研發問題了,而難能可貴的是,在我所接觸的控制演算法當中,PID控制演算法又是最簡單,最能體現反饋思想的控制演算法,可謂經典中的經典。經典的未必是復雜的,經典的東西常常是簡單的,而且是最簡單的,想想牛頓的力學三大定律吧,想想愛因斯坦的質能方程吧,何等的簡單!簡單的不是原始的,簡單的也不是落後的,簡單到了美的程度。 ---【1】
三、PID演算法的C語言源碼:
PID 控制演算法可以分為位置式 PID和增量式 PID控制演算法
詳細見參考【1】【2】
參考:
【1】PID演算法
【2】簡易PID演算法的快速掃盲(超詳細+過程推導+C語言程序)
『肆』 溫度控制的PID演算法的C語言程序
//PID演算法溫控C語言2008-08-17 18:58
#include<reg51.h> 
#include<intrins.h> 
#include<math.h> 
#include<string.h> 
struct PID { 
unsigned int SetPoint; // 設定目標 Desired Value 
unsigned int Proportion; // 比例常數 Proportional Const 
unsigned int Integral; // 積分常數 Integral Const 
unsigned int Derivative; // 微分常數 Derivative Const 
unsigned int LastError; // Error[-1] 
unsigned int PrevError; // Error[-2] 
unsigned int SumError; // Sums of Errors 
}; 
struct PID spid; // PID Control Structure 
unsigned int rout; // PID Response (Output) 
unsigned int rin; // PID Feedback (Input) 
sbit data1=P1^0; 
sbit clk=P1^1; 
sbit plus=P2^0; 
sbit subs=P2^1; 
sbit stop=P2^2; 
sbit output=P3^4; 
sbit DQ=P3^3; 
unsigned char flag,flag_1=0; 
unsigned char high_time,low_time,count=0;//占空比調節參數 
unsigned char set_temper=35; 
unsigned char temper; 
unsigned char i; 
unsigned char j=0; 
unsigned int s; 
/*********************************************************** 
延時子程序,延時時間以12M晶振為准,延時時間為30us×time 
***********************************************************/ 
void delay(unsigned char time) 
{ 
unsigned char m,n; 
for(n=0;n<time;n++) 
for(m=0;m<2;m++){} 
} 
/*********************************************************** 
寫一位數據子程序 
***********************************************************/ 
void write_bit(unsigned char bitval) 
{ 
EA=0; 
DQ=0; /*拉低DQ以開始一個寫時序*/ 
if(bitval==1) 
{ 
_nop_(); 
DQ=1; /*如要寫1,則將匯流排置高*/ 
} 
delay(5); /*延時90us供DA18B20采樣*/ 
DQ=1; /*釋放DQ匯流排*/ 
_nop_(); 
_nop_(); 
EA=1; 
} 
/*********************************************************** 
寫一位元組數據子程序 
***********************************************************/ 
void write_byte(unsigned char val) 
{ 
unsigned char i; 
unsigned char temp; 
EA=0;           /*關中斷*/
TR0=0; 
for(i=0;i<8;i++) /*寫一位元組數據,一次寫一位*/ 
{ 
temp=val>>i; /*移位操作,將本次要寫的位移到最低位*/ 
temp=temp&1; 
write_bit(temp); /*向匯流排寫該位*/ 
} 
delay(7); /*延時120us後*/ 
// TR0=1; 
EA=1; /*開中斷*/
} 
/*********************************************************** 
讀一位數據子程序 
***********************************************************/ 
unsigned char read_bit() 
{ 
unsigned char i,value_bit; 
EA=0; 
DQ=0; /*拉低DQ,開始讀時序*/ 
_nop_(); 
_nop_(); 
DQ=1; /*釋放匯流排*/ 
for(i=0;i<2;i++){} 
value_bit=DQ; 
EA=1; 
return(value_bit); 
} 
/*********************************************************** 
讀一位元組數據子程序 
***********************************************************/ 
unsigned char read_byte() 
{ 
unsigned char i,value=0; 
EA=0; 
for(i=0;i<8;i++) 
{ 
if(read_bit()) /*讀一位元組數據,一個時序中讀一次,並作移位處理*/ 
value|=0x01<<i; 
delay(4); /*延時80us以完成此次都時序,之後再讀下一數據*/ 
} 
EA=1; 
return(value); 
} 
/*********************************************************** 
復位子程序 
***********************************************************/ 
unsigned char reset() 
{ 
unsigned char presence; 
EA=0; 
DQ=0; /*拉低DQ匯流排開始復位*/ 
delay(30); /*保持低電平480us*/ 
DQ=1; /*釋放匯流排*/ 
delay(3); 
presence=DQ; /*獲取應答信號*/ 
delay(28); /*延時以完成整個時序*/ 
EA=1; 
return(presence); /*返回應答信號,有晶元應答返回0,無晶元則返回1*/ 
} 
/*********************************************************** 
獲取溫度子程序 
***********************************************************/ 
void get_temper() 
{ 
unsigned char i,j; 
do 
{ 
i=reset(); /*復位*/ 
}while(i!=0); /*1為無反饋信號*/ 
i=0xcc; /*發送設備定位命令*/ 
write_byte(i); 
i=0x44; /*發送開始轉換命令*/ 
write_byte(i); 
delay(180); /*延時*/ 
do 
{ 
i=reset(); /*復位*/ 
}while(i!=0); 
i=0xcc; /*設備定位*/ 
write_byte(i); 
i=0xbe; /*讀出緩沖區內容*/ 
write_byte(i); 
j=read_byte(); 
i=read_byte(); 
i=(i<<4)&0x7f; 
s=(unsigned int)(j&0x0f); 
s=(s*100)/16; 
j=j>>4; 
temper=i|j; /*獲取的溫度放在temper中*/ 
} 
/*==================================================================================================== 
Initialize PID Structure 
=====================================================================================================*/ 
void PIDInit (struct PID *pp) 
{ 
memset ( pp,0,sizeof(struct PID)); 
} 
/*==================================================================================================== 
PID計算部分 
=====================================================================================================*/ 
unsigned int PIDCalc( struct PID *pp, unsigned int NextPoint ) 
{ 
unsigned int dError,Error; 
Error = pp->SetPoint - NextPoint; // 偏差 
pp->SumError += Error; // 積分 
dError = pp->LastError - pp->PrevError; // 當前微分 
pp->PrevError = pp->LastError; 
pp->LastError = Error; 
return (pp->Proportion * Error//比例
+ pp->Integral * pp->SumError  //積分項
+ pp->Derivative * dError); //   微分項 
} 
/*********************************************************** 
溫度比較處理子程序 
***********************************************************/ 
compare_temper() 
{ 
unsigned char i; 
if(set_temper>temper) 
{ 
if(set_temper-temper>1) 
{ 
high_time=100; 
low_time=0; 
} 
else 
{ 
for(i=0;i<10;i++) 
{ get_temper(); 
rin = s; // Read Input 
rout = PIDCalc ( &spid,rin ); // Perform PID Interation 
} 
if (high_time<=100) 
high_time=(unsigned char)(rout/800); 
else 
high_time=100; 
low_time= (100-high_time); 
} 
} 
else if(set_temper<=temper) 
{ 
if(temper-set_temper>0) 
{ 
high_time=0; 
low_time=100; 
} 
else 
{ 
for(i=0;i<10;i++) 
{ get_temper(); 
rin = s; // Read Input 
rout = PIDCalc ( &spid,rin ); // Perform PID Interation 
} 
if (high_time<100) 
high_time=(unsigned char)(rout/10000); 
else 
high_time=0; 
low_time= (100-high_time); 
} 
} 
// else 
// {} 
} 
/***************************************************** 
T0中斷服務子程序,用於控制電平的翻轉 ,40us*100=4ms周期 
******************************************************/ 
void serve_T0() interrupt 1 using 1 
{ 
if(++count<=(high_time)) 
output=1; 
else if(count<=100) 
{ 
output=0; 
} 
else 
count=0; 
TH0=0x2f; 
TL0=0xe0; 
} 
/***************************************************** 
串列口中斷服務程序,用於上位機通訊 
******************************************************/ 
void serve_sio() interrupt 4 using 2 
{ 
/* EA=0; 
RI=0; 
i=SBUF; 
if(i==2) 
{ 
while(RI==0){} 
RI=0; 
set_temper=SBUF; 
SBUF=0x02; 
while(TI==0){} 
TI=0; 
} 
else if(i==3) 
{ 
TI=0; 
SBUF=temper; 
while(TI==0){} 
TI=0; 
} 
EA=1; */ 
} 
void disp_1(unsigned char disp_num1[6]) 
{ 
unsigned char n,a,m; 
for(n=0;n<6;n++) 
{ 
// k=disp_num1[n]; 
for(a=0;a<8;a++) 
{ 
clk=0; 
m=(disp_num1[n]&1); 
disp_num1[n]=disp_num1[n]>>1; 
if(m==1) 
data1=1; 
else 
data1=0; 
_nop_(); 
clk=1; 
_nop_(); 
} 
} 
} 
/***************************************************** 
顯示子程序 
功能:將占空比溫度轉化為單個字元,顯示占空比和測得到的溫度 
******************************************************/ 
void display() 
{ 
unsigned char code number[]={0xfc,0x60,0xda,0xf2,0x66,0xb6,0xbe,0xe0,0xfe,0xf6}; 
unsigned char disp_num[6]; 
unsigned int k,k1; 
k=high_time; 
k=k%1000; 
k1=k/100; 
if(k1==0) 
disp_num[0]=0; 
else 
disp_num[0]=0x60; 
k=k%100; 
disp_num[1]=number[k/10]; 
disp_num[2]=number[k%10]; 
k=temper; 
k=k%100; 
disp_num[3]=number[k/10]; 
disp_num[4]=number[k%10]+1; 
disp_num[5]=number[s/10]; 
disp_1(disp_num); 
} 
/*********************************************************** 
主程序 
***********************************************************/ 
main() 
{ 
unsigned char z; 
unsigned char a,b,flag_2=1,count1=0; 
unsigned char phil[]={2,0xce,0x6e,0x60,0x1c,2}; 
TMOD=0x21; 
TH0=0x2f; 
TL0=0x40; 
SCON=0x50; 
PCON=0x00; 
TH1=0xfd; 
TL1=0xfd; 
PS=1; 
EA=1; 
EX1=0; 
ET0=1; 
ES=1; 
TR0=1; 
TR1=1; 
high_time=50; 
low_time=50; 
PIDInit ( &spid ); // Initialize Structure 
spid.Proportion = 10; // Set PID Coefficients 
spid.Integral = 8; 
spid.Derivative =6; 
spid.SetPoint = 100; // Set PID Setpoint 
while(1) 
{ 
if(plus==0) 
{ 
EA=0; 
for(a=0;a<5;a++) 
for(b=0;b<102;b++){} 
if(plus==0) 
{ 
set_temper++; 
flag=0; 
} 
} 
else if(subs==0) 
{ 
for(a=0;a<5;a++) 
for(b=0;a<102;b++){} 
if(subs==0) 
{ 
set_temper--; 
flag=0; 
} 
} 
else if(stop==0) 
{ 
for(a=0;a<5;a++) 
for(b=0;b<102;b++){} 
if(stop==0) 
{ 
flag=0; 
break; 
} 
EA=1; 
} 
get_temper(); 
b=temper; 
if(flag_2==1) 
a=b; 
if((abs(a-b))>5) 
temper=a; 
else 
temper=b; 
a=temper; 
flag_2=0; 
if(++count1>30) 
{ 
display(); 
count1=0; 
} 
compare_temper(); 
} 
TR0=0; 
z=1; 
while(1) 
{ 
EA=0; 
if(stop==0) 
{ 
for(a=0;a<5;a++) 
for(b=0;b<102;b++){} 
if(stop==0) 
disp_1(phil); 
// break; 
} 
EA=1; 
} 
} 
//DS18b20 子程序
#include <REG52.H>
sbit DQ=P2^1;              //定義埠
typedef unsigned char  byte;
typedef unsigned int   word;
                             //延時
void delay(word useconds)
{
   for(;useconds>0;useconds--);
}
                             //復位
byte ow_reset(void)
{
  byte presence;
  DQ=0;                     //DQ低電平
  delay(29);                //480us
  DQ=1;                     //DQ高電平
  delay(3);                 //等待
  presence=DQ;             //presence信號
  delay(25);
  return(presence);
}                          //0允許,1禁止
                           //從1-wire  匯流排上讀取一個位元組
byte read_byte(viod)
{
   byte i;
   byte value=0;
   for (i=8;i>0;i--)
   {
     value>>=1;
     DQ=0;
     DQ=1;
     delay(1);
     
     if(DQ)value|=0x80;
     delay(6);
    }
    return(value);
}
                              //向1-wire匯流排上寫一個位元組
void write_byte(char val)
{ 
  byte  i;
  for (i=8;i>0;i--)           //一次寫一個位元組
  {
   DQ=0;
   DQ=val&0x01;
   delay(5);
   DQ=1;
   val=val/2;
  }
   delay(5);
 }
                              //讀取溫度
char Read_Temperature(void)
{
   union{
      byte c[2];
      int x;
   }temp;
   ow_reset();
   write_byte(0xcc);
   write_byte(0xBE);
   temp.c[1]=read_byte();
   temp.c[0]=read_byte();
   ow_reset();
   write_byte(0xCC);
   write_byte(0x44);
   return  temp.x/2;
}
『伍』 C語言編程,怎麼計算PID控制
Step 7寫的PID控制的FC模塊。帶"_IN"與帶"_OUT"的變數,如果前綴是一樣的,要求連接同一個變數。
FUNCTION FC1 : VOID
 VAR_INPUT
 Run:BOOL; //True-運行,False-停止
 Auto:BOOL; //True-自動,False-手動
 ISW:BOOL; //True-積分有效,False-積分無效
 DSW:BOOL; //True-微分有效,False-微分無效
 SetMV:REAL; //手動時的開度設定值
 SVSW:REAL; //當設定值低於SVSW時,開度為零
 PV:REAL; //測量值
 SV:REAL; //設定值
 DeadBand:REAL; //死區大小
 PBW:REAL; //比例帶大小
 IW:REAL; //積分帶大小
 DW:REAL; //微分帶大小
 dErr_IN:REAL; //誤差累積
 LastPV_IN:REAL; //上一控制周期的測量值
 END_VAR
 VAR_OUTPUT
 MV:REAL; //輸出開度
 dErr_OUT:REAL; //誤差累積
 LastPV_OUT:REAL;//上一控制周期的測量值
 END_VAR
 VAR
 Err:REAL; //誤差
 dErr:REAL; //誤差累積
 PBH:REAL; //比例帶上限
 PBL:REAL; //比例帶下限
 PVC:REAL; //測量值在一個控制周期內的變化率,即測量值變化速率
 P:REAL; //比例項
 I:REAL; //積分項
 D:REAL; //微分項
 END_VAR
 IF Run=1 THEN
 IF Auto=1 THEN
 IF SV>=SVSW THEN
 Err:=SV-PV;
 PBH:=SV+PBW;
 PBL:=SV-PBW;
 IF PV<PBL THEN
 MV:=1;
 ELSIF PV>PBH THEN
 MV:=0;
 ELSE
 P:=(PBH-PV)/(PBH-PBL); //計算比例項
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
/////////////////////////////////////////////以下為積分項的計算//////////////////////////////////////////////////////////////
 IF ISW=1 THEN
 dErr:=dErr_IN;
 IF (PV<(SV-DeadBand)) OR (PV>(SV+DeadBand)) THEN
 IF (dErr+Err)<(0-IW) THEN
 dErr:=0-IW;
 ELSIF (dErr+Err)>IW THEN
 dErr:=IW;
 ELSE
 dErr:=dErr+Err;
 END_IF;
 END_IF;
 I:=dErr/IW;
 dErr_OUT:=dErr;
 ELSE
 I:=0;
 END_IF;
/////////////////////////////////////////////以上為積分項的計算//////////////////////////////////////////////////////////////
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
/////////////////////////////////////////////以下為微分項的計算//////////////////////////////////////////////////////////////
 IF DSW=1 THEN
 PVC:=LastPV_IN-PV;
 D:=PVC/DW;
 LastPV_OUT:=PV;
 ELSE
 D:=0;
 END_IF;
/////////////////////////////////////////////以上為微分項的計算//////////////////////////////////////////////////////////////
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
 IF (P+I+D)>1 THEN
 MV:=1;
 ELSIF (P+I+D)<0 THEN
 MV:=0;
 ELSE
 MV:=P+I+D;
 END_IF;
 END_IF;
 ELSE
 MV:=0;
 END_IF;
 ELSE
 MV:=SetMV;
 END_IF;
 ELSE
 MV:=0;
 END_IF;
END_FUNCTION
進行整定時先進行P調節,使I和D作用無效,觀察溫度變化曲線,若變化曲線多次出現波形則應該放大比例(P)參數,若變化曲線非常平緩,則應該縮小比例(P)參數。比例(P)參數設定好後,設定積分(I)參數,積分(I)正好與P參數相反,曲線平緩則需要放大積分(I),出現多次波形則需要縮小積分(I)。比例(P)和積分(I)都設定好以後設定微分(D)參數,微分(D)參數與比例(P)參數的設定方法是一樣的。 
當初寫這段程序的就是為了使用調功器來控制爐子的溫度的,已經在我單位的調功器上運行成功了,還有就是我單位的調功器沒有使用微分(D),只是用了比例(P)和積分(I)。
『陸』 求一下增量式和位置式PID的C語言程序
增量式PID:
typedefstruct{
floatscope;//輸出限幅量
floataim;//目標輸出量
floatreal_out;//實際輸出量
floatKp;
floatKi;
floatKd;
floate0;//當前誤差
floate1;//上一次誤差
floate2;//上上次誤差
}PID_Type;
#definemin(a,b)(a<b?a:b)
#definemax(a,b)(a>b?a:b)
#definelimiter(x,a,b)(min(max(x,a),b))
#defineexchange(a,b,tmp)(tmp=a,a=b,b=tmp)
#definemyabs(x)((x<0)?-x:x)
floatpid_acc(PID_Type*pid)
{
floatout;
floatep,ei,ed;
pid->e0=pid->aim-pid->real_out;
ep=pid->e0-pid->e1;
ei=pid->e0;
ed=pid->e0-2*pid->e1+pid->e2;
out=pid->Kp*ep+pid->Ki*ei+pid->Kd*ed;
out=limiter(out,-pid->scope,pid->scope);
pid->e2=pid->e1;
pid->e1=pid->e0;
returnout;
}位置式PID:
typedefstruct{
floatscope;//輸出限幅量
floataim;//目標輸出量
floatreal_out;//反饋輸出量
floatKp;
floatKi;
floatKd;
floatSum;
floate0;//當前誤差
floate1;//上一次誤差
}PID_Type;
#definemax(a,b)(a>b?a:b)
#definemin(a,b)(a<b?a:b)
#definelimiter(x,a,b)(min(max(x,a),b))
floatpid_pos(PID_Type*p)
{
floatpe,ie,de;
floatout=0;
p->e0=p->aim-p->real_out;//計算當前誤差
p->Sum+=p->e0;//誤差積分
de=p->e0-p->e1;//誤差微分
pe=p->e0;
ie=p->Sum;
p->e1=p->e0;
out=pe*(p->Kp)+ie*(p->Ki)+de*(p->Kd);
out=limiter(out,-p->scope,p->scope);//輸出限幅
returnout;
}親手移植到我的stm32小車上 調試3個參數後正常使用。
『柒』 高精度的恆溫控制PID程序,8位MCU用C語言
以下程序供參考:
//51 溫度PID經典演算法
#include<reg51.h>  
#include<intrins.h>  
#include<math.h>  
#include<string.h>  
struct PID {  
unsigned int SetPoint; // 設定目標 Desired Value  
unsigned int Proportion; // 比例常數 Proportional Const  
unsigned int Integral; // 積分常數 Integral Const  
unsigned int Derivative; // 微分常數 Derivative Const  
unsigned int LastError; // Error[-1]  
unsigned int PrevError; // Error[-2]  
unsigned int SumError; // Sums of Errors  
};  
struct PID spid; // PID Control Structure  
unsigned int rout; // PID Response (Output)  
unsigned int rin; // PID Feedback (Input)  
sbit data1=P1^0;  
sbit clk=P1^1;  
sbit plus=P2^0;  
sbit subs=P2^1;  
sbit stop=P2^2;  
sbit output=P3^4;  
sbit DQ=P3^3;  
unsigned char flag,flag_1=0;  
unsigned char high_time,low_time,count=0;//占空比調節參數  
unsigned char set_temper=25;  
unsigned char temper;  
unsigned char i;  
unsigned char j=0;  
unsigned int s;  
/***********************************************************  
延時子程序,延時時間以12M晶振為准,延時時間為30us×time  
***********************************************************/  
void delay(unsigned char time)  
{  
     unsigned char m,n;  
     for(n=0;n<time;n++)  
     for(m=0;m<2;m++){}  
}  
/***********************************************************  
寫一位數據子程序  
***********************************************************/  
void write_bit(unsigned char bitval)  
{  
  EA=0;  
  DQ=0; /*拉低DQ以開始一個寫時序*/  
if(bitval==1)  
{  
  _nop_();  
  DQ=1; /*如要寫1,則將匯流排置高*/  
}  
 delay(5); /*延時90us供DA18B20采樣*/  
 DQ=1; /*釋放DQ匯流排*/  
_nop_();  
_nop_();  
EA=1;  
}  
/***********************************************************  
寫一位元組數據子程序  
***********************************************************/  
void write_byte(unsigned char val)  
{  
     unsigned char i;  
    unsigned char temp;  
    EA=0;  
    TR0=0;  
for(i=0;i<8;i++) /*寫一位元組數據,一次寫一位*/  
{  
  temp=val>>i; /*移位操作,將本次要寫的位移到最低位*/  
  temp=temp&1;  
  write_bit(temp); /*向匯流排寫該位*/  
}  
  delay(7); /*延時120us後*/  
// TR0=1;  
  EA=1;  
}  
/***********************************************************  
讀一位數據子程序  
***********************************************************/  
unsigned char read_bit()  
{  
unsigned char i,value_bit;  
EA=0;  
DQ=0; /*拉低DQ,開始讀時序*/  
_nop_();  
_nop_();  
DQ=1; /*釋放匯流排*/  
for(i=0;i<2;i++){}  
value_bit=DQ;  
EA=1;  
return(value_bit);  
}  
/***********************************************************  
讀一位元組數據子程序  
***********************************************************/  
unsigned char read_byte()  
{  
unsigned char i,value=0;  
EA=0;  
for(i=0;i<8;i++)  
{  
if(read_bit()) /*讀一位元組數據,一個時序中讀一次,並作移位處理*/  
value|=0x01<<i;  
delay(4); /*延時80us以完成此次都時序,之後再讀下一數據*/  
}  
EA=1;  
return(value);  
}  
/***********************************************************  
復位子程序  
***********************************************************/  
unsigned char reset()  
{  
unsigned char presence;  
EA=0;  
DQ=0; /*拉低DQ匯流排開始復位*/  
delay(30); /*保持低電平480us*/  
DQ=1; /*釋放匯流排*/  
delay(3);  
presence=DQ; /*獲取應答信號*/  
delay(28); /*延時以完成整個時序*/  
EA=1;  
return(presence); /*返回應答信號,有晶元應答返回0,無晶元則返回1*/  
}  
/***********************************************************  
獲取溫度子程序  
***********************************************************/  
void get_temper()  
{  
unsigned char i,j;  
do  
{  
   i=reset(); /*復位*/  
}  while(i!=0); /*1為無反饋信號*/  
     i=0xcc; /*發送設備定位命令*/  
   write_byte(i);  
   i=0x44; /*發送開始轉換命令*/  
   write_byte(i);  
   delay(180); /*延時*/  
do  
{  
   i=reset(); /*復位*/  
}  while(i!=0);  
   i=0xcc; /*設備定位*/  
   write_byte(i);  
   i=0xbe; /*讀出緩沖區內容*/  
   write_byte(i);  
   j=read_byte();    
   i=read_byte();  
   i=(i<<4)&0x7f;  
   s=(unsigned int)(j&0x0f);     //得到小數部分 
   s=(s*100)/16;  
   j=j>>4;  
   temper=i|j; /*獲取的溫度放在temper中*/  
}  
/*====================================================================================================  
Initialize PID Structure  
=====================================================================================================*/  
void PIDInit (struct PID *pp)  
{  
memset ( pp,0,sizeof(struct PID));    //全部初始化為0 
}  
/*====================================================================================================  
PID計算部分  
=====================================================================================================*/  
unsigned int PIDCalc( struct PID *pp, unsigned int NextPoint )  
{  
unsigned int dError,Error;  
Error = pp->SetPoint - NextPoint;          // 偏差     
pp->SumError += Error;                     // 積分     
dError = pp->LastError - pp->PrevError;    // 當前微分   
pp->PrevError = pp->LastError;     
pp->LastError = Error;  
return (pp->Proportion * Error             // 比例項     
+ pp->Integral * pp->SumError              // 積分項  
+ pp->Derivative * dError);                // 微分項  
}  
/***********************************************************  
溫度比較處理子程序  
***********************************************************/  
void compare_temper()  
{  
unsigned char i;  
if(set_temper>temper)      //是否設置的溫度大於實際溫度 
{  
   if(set_temper-temper>1)  //設置的溫度比實際的溫度是否是大於1度 
  {  
   high_time=100;      //如果是,則全速加熱 
   low_time=0;  
  }  
       else  //如果是在1度范圍內,則運行PID計算 
  {  
    for(i=0;i<10;i++)  
  {  
    get_temper();   //獲取溫度 
     rin = s; // Read Input  
    rout = PIDCalc ( &spid,rin ); // Perform PID Interation  
  }  
    if (high_time<=100)  
      high_time=(unsigned char)(rout/800);  
    else  
          high_time=100;  
      low_time= (100-high_time);  
  }  
}  
else if(set_temper<=temper)  
{  
   if(temper-set_temper>0)  
  {  
    high_time=0;  
    low_time=100;  
  }  
   else  
  {  
     for(i=0;i<10;i++)  
   { 
         get_temper();  
         rin = s; // Read Input  
     rout = PIDCalc ( &spid,rin ); // Perform PID Interation  
   }  
     if (high_time<100)  
      high_time=(unsigned char)(rout/10000);  
       else  
      high_time=0;  
      low_time= (100-high_time);  
  }  
}  
// else  
// {}  
}  
/*****************************************************  
T0中斷服務子程序,用於控制電平的翻轉 ,40us*100=4ms周期  
******************************************************/  
void serve_T0() interrupt 1 using 1  
{  
if(++count<=(high_time))  
output=1;  
else if(count<=100)  
{  
output=0;  
}  
else  
count=0;  
TH0=0x2f;  
TL0=0xe0;  
}  
/*****************************************************  
串列口中斷服務程序,用於上位機通訊  
******************************************************/  
void serve_sio() interrupt 4 using 2  
{  
/* EA=0;  
RI=0;  
i=SBUF;  
if(i==2)  
{  
while(RI==0){}  
RI=0;  
set_temper=SBUF;  
SBUF=0x02;  
while(TI==0){}  
TI=0;  
}  
else if(i==3)  
{  
TI=0;  
SBUF=temper;  
while(TI==0){}  
TI=0;  
}  
EA=1; */  
}  
void disp_1(unsigned char disp_num1[6])  
{  
unsigned char n,a,m;  
for(n=0;n<6;n++)  
{  
// k=disp_num1[n];  
 for(a=0;a<8;a++)  
 {  
     clk=0;  
  m=(disp_num1[n]&1);  
  disp_num1[n]=disp_num1[n]>>1;  
  if(m==1)  
   data1=1;  
  else  
   data1=0;  
   _nop_();  
   clk=1;  
   _nop_();  
 }  
}  
}  
/*****************************************************  
顯示子程序  
功能:將占空比溫度轉化為單個字元,顯示占空比和測得到的溫度  
******************************************************/  
void display()  
{  
unsigned char code number[]={0xfc,0x60,0xda,0xf2,0x66,0xb6,0xbe,0xe0,0xfe,0xf6};  
unsigned char disp_num[6];  
unsigned int k,k1;  
k=high_time;  
k=k%1000;  
k1=k/100;  
if(k1==0)  
disp_num[0]=0;  
else  
disp_num[0]=0x60;  
k=k%100;  
disp_num[1]=number[k/10];  
disp_num[2]=number[k%10];  
k=temper;  
k=k%100;  
disp_num[3]=number[k/10];  
disp_num[4]=number[k%10]+1;  
disp_num[5]=number[s/10];  
disp_1(disp_num);  
}  
/***********************************************************  
主程序  
***********************************************************/  
void main()  
{  
unsigned char z; 
unsigned char a,b,flag_2=1,count1=0;  
unsigned char phil[]={2,0xce,0x6e,0x60,0x1c,2}; 
TMOD=0x21;  
TH0=0x2f;  
TL0=0x40;  
SCON=0x50;  
PCON=0x00;  
TH1=0xfd;  
TL1=0xfd;  
PS=1;  
EA=1;  
EX1=0;  
ET0=1;  
ES=1;  
TR0=1;  
TR1=1;  
high_time=50;  
low_time=50;  
PIDInit ( &spid );    // Initialize Structure  
spid.Proportion = 10; // Set PID Coefficients  比例常數 Proportional Const  
spid.Integral = 8;    //積分常數 Integral Const  
spid.Derivative =6;   //微分常數 Derivative Const  
spid.SetPoint = 100; // Set PID Setpoint 設定目標 Desired Value  
while(1)  
 {  
if(plus==0)  
 { 
EA=0;  
for(a=0;a<5;a++)  
for(b=0;b<102;b++){}  
if(plus==0)  
  { 
set_temper++;  
flag=0;  
  } 
 }  
else if(subs==0)  
  {  
for(a=0;a<5;a++)  
for(b=0;a<102;b++){}  
if(subs==0)  
{  
 set_temper--;  
 flag=0;  
}  
  }  
else if(stop==0)  
{  
     for(a=0;a<5;a++)  
    for(b=0;b<102;b++){}  
    if(stop==0)  
{  
   flag=0;  
   break;  
}  
   EA=1;  
}  
       get_temper();  
   b=temper;  
if(flag_2==1)  
  a=b;  
if((abs(a-b))>5)  
  temper=a;  
else  
  temper=b;  
  a=temper;  
  flag_2=0;  
if(++count1>30)  
{  
  display();  
  count1=0;  
}  
  compare_temper();  
}  
   TR0=0;  
   z=1;  
while(1)  
{  
    EA=0;  
if(stop==0)  
{  
     for(a=0;a<5;a++)  
    for(b=0;b<102;b++){}  
    if(stop==0)  
    disp_1(phil);  
// break;  
}  
EA=1;  
}  
}
『捌』 PID演算法溫控C語言
1. PID調試步驟 
沒有一種控制演算法比PID調節規律更有效、更方便的了。現在一些時髦點的調節器基本源自PID。甚至可以這樣說:PID調節器是其它控制調節演算法的媽。 
為什麼PID應用如此廣泛、又長久不衰? 
因為PID解決了自動控制理論所要解決的最基本問題,既系統的穩定性、快速性和准確性。調節PID的參數,可實現在系統穩定的前提下,兼顧系統的帶載能力和抗擾能力,同時,在PID調節器中引入積分項,系統增加了一個零積點,使之成為一階或一階以上的系統,這樣系統階躍響應的穩態誤差就為零。 
由於自動控制系統被控對象的千差萬別,PID的參數也必須隨之變化,以滿足系統的性能要求。這就給使用者帶來相當的麻煩,特別是對初學者。下面簡單介紹一下調試PID參數的一般步驟: 
1.負反饋 
自動控制理論也被稱為負反饋控制理論。首先檢查系統接線,確定系統的反饋為負反饋。例如電機調速系統,輸入信號為正,要求電機正轉時,反饋信號也為正(PID演算法時,誤差=輸入-反饋),同時電機轉速越高,反饋信號越大。其餘系統同此方法。 
2.PID調試一般原則 
a.在輸出不振盪時,增大比例增益P。 
b.在輸出不振盪時,減小積分時間常數Ti。 
c.在輸出不振盪時,增大微分時間常數Td。 
3.一般步驟 
a.確定比例增益P 
確定比例增益P 時,首先去掉PID的積分項和微分項,一般是令Ti=0、Td=0(具體見PID的參數設定說明),使PID為純比例調節。輸入設定為系統允許的最大值的60%~70%,由0逐漸加大比例增益P,直至系統出現振盪;再反過來,從此時的比例增益P逐漸減小,直至系統振盪消失,記錄此時的比例增益P,設定PID的比例增益P為當前值的60%~70%。比例增益P調試完成。 
b.確定積分時間常數Ti 
比例增益P確定後,設定一個較大的積分時間常數Ti的初值,然後逐漸減小Ti,直至系統出現振盪,之後在反過來,逐漸加大Ti,直至系統振盪消失。記錄此時的Ti,設定PID的積分時間常數Ti為當前值的150%~180%。積分時間常數Ti調試完成。 
c.確定積分時間常數Td 
積分時間常數Td一般不用設定,為0即可。若要設定,與確定 P和Ti的方法相同,取不振盪時的30%。 
d.系統空載、帶載聯調,再對PID參數進行微調,直至滿足要求。 
2.PID控制簡介 
目前工業自動化水平已成為衡量各行各業現代化水平的一個重要標志。同時,控制理論的發展也經歷了古典控制理論、現代控制理論和智能控制理論三個階段。智能控制的典型實例是模糊全自動洗衣機等。自動控制系統可分為開環控制系統和閉環控制系統。一個控制系統包括控制器、感測器、變送器、執行機構、輸入輸出介面。控制器的輸出經過輸出介面、執行機構,加到被控系統上;控制系統的被控量,經過感測器,變送器,通過輸入介面送到控制器。不同的控制系統,其感測器、變送器、執行機構是不一樣的。比如壓力控制系統要採用壓力感測器。電加熱控制系統的感測器是溫度感測器。目前,PID控制及其控制器或智能PID控制器(儀表)已經很多,產品已在工程實際中得到了廣泛的應用,有各種各樣的PID控制器產品,各大公司均開發了具有PID參數自整定功能的智能調節器(intelligent regulator),其中PID控制器參數的自動調整是通過智能化調整或自校正、自適應演算法來實現。有利用PID控制實現的壓力、溫度、流量、液位控制器,能實現PID控制功能的可編程式控制制器(PLC),還有可實現PID控制的PC系統等等。 可編程式控制制器(PLC) 是利用其閉環控制模塊來實現PID控制,而可編程式控制制器(PLC)可以直接與ControlNet相連,如Rockwell的PLC-5等。還有可以實現PID控制功能的控制器,如Rockwell 的Logix產品系列,它可以直接與ControlNet相連,利用網路來實現其遠程式控制制功能。 
1、開環控制系統 
開環控制系統(open-loop control system)是指被控對象的輸出(被控制量)對控制器(controller)的輸出沒有影響。在這種控制系統中,不依賴將被控量反送回來以形成任何閉環迴路。 
2、閉環控制系統 
閉環控制系統(closed-loop control system)的特點是系統被控對象的輸出(被控制量)會反送回來影響控制器的輸出,形成一個或多個閉環。閉環控制系統有正反饋和負反饋,若反饋信號與系統給定值信號相反,則稱為負反饋( Negative Feedback),若極性相同,則稱為正反饋,一般閉環控制系統均採用負反饋,又稱負反饋控制系統。閉環控制系統的例子很多。比如人就是一個具有負反饋的閉環控制系統,眼睛便是感測器,充當反饋,人體系統能通過不斷的修正最後作出各種正確的動作。如果沒有眼睛,就沒有了反饋迴路,也就成了一個開環控制系統。另例,當一台真正的全自動洗衣機具有能連續檢查衣物是否洗凈,並在洗凈之後能自動切斷電源,它就是一個閉環控制系統。 
3、階躍響應 
階躍響應是指將一個階躍輸入(step function)加到系統上時,系統的輸出。穩態誤差是指系統的響應進入穩態後,系統的期望輸出與實際輸出之差。控制系統的性能可以用穩、准、快三個字來描述。穩是指系統的穩定性(stability),一個系統要能正常工作,首先必須是穩定的,從階躍響應上看應該是收斂的;準是指控制系統的准確性、控制精度,通常用穩態誤差來(Steady-state error)描述,它表示系統輸出穩態值與期望值之差;快是指控制系統響應的快速性,通常用上升時間來定量描述。 
4、PID控制的原理和特點 
在工程實際中,應用最為廣泛的調節器控制規律為比例、積分、微分控制,簡稱PID控制,又稱PID調節。PID控制器問世至今已有近70年歷史,它以其結構簡單、穩定性好、工作可靠、調整方便而成為工業控制的主要技術之一。當被控對象的結構和參數不能完全掌握,或得不到精確的數學模型時,控制理論的其它技術難以採用時,系統控制器的結構和參數必須依靠經驗和現場調試來確定,這時應用PID控制技術最為方便。即當我們不完全了解一個系統和被控對象,或不能通過有效的測量手段來獲得系統參數時,最適合用PID控制技術。PID控制,實際中也有PI和PD控制。PID控制器就是根據系統的誤差,利用比例、積分、微分計算出控制量進行控制的。 
比例(P)控制 
比例控制是一種最簡單的控制方式。其控制器的輸出與輸入誤差信號成比例關系。當僅有比例控制時系統輸出存在穩態誤差(Steady-state error)。 
積分(I)控制 
在積分控制中,控制器的輸出與輸入誤差信號的積分成正比關系。對一個自動控制系統,如果在進入穩態後存在穩態誤差,則稱這個控制系統是有穩態誤差的或簡稱有差系統(System with Steady-state Error)。為了消除穩態誤差,在控制器中必須引入「積分項」。積分項對誤差取決於時間的積分,隨著時間的增加,積分項會增大。這樣,即便誤差很小,積分項也會隨著時間的增加而加大,它推動控制器的輸出增大使穩態誤差進一步減小,直到等於零。因此,比例+積分(PI)控制器,可以使系統在進入穩態後無穩態誤差。 
微分(D)控制 
在微分控制中,控制器的輸出與輸入誤差信號的微分(即誤差的變化率)成正比關系。 自動控制系統在克服誤差的調節過程中可能會出現振盪甚至失穩。其原因是由於存在有較大慣性組件(環節)或有滯後(delay)組件,具有抑制誤差的作用,其變化總是落後於誤差的變化。解決的辦法是使抑制誤差的作用的變化「超前」,即在誤差接近零時,抑制誤差的作用就應該是零。這就是說,在控制器中僅引入「比例」項往往是不夠的,比例項的作用僅是放大誤差的幅值,而目前需要增加的是「微分項」,它能預測誤差變化的趨勢,這樣,具有比例+微分的控制器,就能夠提前使抑制誤差的控製作用等於零,甚至為負值,從而避免了被控量的嚴重超調。所以對有較大慣性或滯後的被控對象,比例+微分(PD)控制器能改善系統在調節過程中的動態特性。 
5、PID控制器的參數整定 
PID控制器的參數整定是控制系統設計的核心內容。它是根據被控過程的特性確定PID控制器的比例系數、積分時間和微分時間的大小。PID控制器參數整定的方法很多,概括起來有兩大類:一是理論計算整定法。它主要是依據系統的數學模型,經過理論計算確定控制器參數。這種方法所得到的計算數據未必可以直接用,還必須通過工程實際進行調整和修改。二是工程整定方法,它主要依賴工程經驗,直接在控制系統的試驗中進行,且方法簡單、易於掌握,在工程實際中被廣泛採用。PID控制器參數的工程整定方法,主要有臨界比例法、反應曲線法和衰減法。三種方法各有其特點,其共同點都是通過試驗,然後按照工程經驗公式對控制器參數進行整定。但無論採用哪一種方法所得到的控制器參數,都需要在實際運行中進行最後調整與完善。現在一般採用的是臨界比例法。利用該方法進行 PID控制器參數的整定步驟如下:(1)首先預選擇一個足夠短的采樣周期讓系統工作;(2)僅加入比例控制環節,直到系統對輸入的階躍響應出現臨界振盪,記下這時的比例放大系數和臨界振盪周期;(3)在一定的控制度下通過公式計算得到PID控制器的參數。 
3.PID控制器參數的工程整定,各種調節系統中PID參數經驗數據以下可參照: 
溫度T: P=20~60%,T=180~600s,D=3-180s 
壓力P: P=30~70%,T=24~180s, 
液位L: P=20~80%,T=60~300s, 
流量L: P=40~100%,T=6~60s。 
4. PID常用口訣: 
參數整定找最佳,從小到大順序查 
先是比例後積分,最後再把微分加 
曲線振盪很頻繁,比例度盤要放大 
曲線漂浮繞大灣,比例度盤往小扳 
曲線偏離回復慢,積分時間往下降 
曲線波動周期長,積分時間再加長 
曲線振盪頻率快,先把微分降下來 
動差大來波動慢。微分時間應加長 
理想曲線兩個波,前高後低4比1 
一看二調多分析,調節質量不會低
參考資料:http://www.yuanqijian.com/bbs/htmled_topic.php?topi_id=64489
『玖』 pid控制的C語言編程
#include<unistd.h>
#include<stdio.h>
int main(int argc,int **argv)
{
int pid=fork();
if(pid==-1)
{
printf("error");
}
else if(pid==0)
{
printf("This is the child process!\n");
}
else
{
printf("This is the parent process! child process id=%d\n",pid);
}
return 0;
}
首先為什麼這段代碼gcc編譯不了,只能用g++編譯,gcc編譯顯示結果如下
Undefined                       first referenced
 symbol                             in file
__gxx_personality_v0                /var/tmp//ccuHN8IS.o
ld: fatal: Symbol referencing errors. No output written to t5
collect2: ld returned 1 exit status
其次,g++編譯後運行結果如下
This is the parent process! child process id=27406
This is the child process!
『拾』 緊急求助單片機溫控系統思路,光耦控制可控硅,PID演算法
一、單向可控硅是一種可控整流電子元件,能在外部控制信號作用下由關斷變為導通,但一旦導通,外部信號就無法使其關斷,只能靠去除負載或降低其兩端電壓使其關斷。單向可控硅是由三個PN結PNPN組成的四層三端半導體器件與具有一個PN結的二極體相比,單向可控硅正向導通受控制極電流控制;與具有兩個PN結的三極體相比,差別在於可控硅對控制極電流沒有放大作用。
  二、雙向可控硅具有兩個方向輪流導通、關斷的特性。雙向可控硅實質上是兩個反並聯的單向可控硅,是由NPNPN五層半導體形成四個PN結構成、有三個電極的半導體器件。由於主電極的構造是對稱的(都從N層引出),所以它的電極不像單向可控硅那樣分別叫陽極和陰極,而是把與控制極相近的叫做第一電極A1,另一個叫做第二電極A2。雙向可控硅的主要缺點是承受電壓上升率的能力較低。這是因為雙向可控硅在一個方向導通結束時,矽片在各層中的載流子還沒有回到截止狀態的位置,必須採取相應的保護措施。雙向可控硅元件主要用於交流控制電路,如溫度控制、燈光控制、防爆交流開關以及直流電機調速和換向等電路。
  單向可控硅和雙向可控硅,都是三個電極。單向可控硅有陰極(K)、陽極(A)、控制極(G)。雙向可控硅等效於兩只單項可控硅反向並聯而成。即其中一隻單向硅陽極與另一隻陰極相邊連,其引出端稱T2極,其中一隻單向硅陰極與另一隻陽極相連,其引出端稱T2極,剩下則為控制極(G)。
  1、單、雙向可控硅的判別:先任測兩個極,若正、反測指針均不動(R×1擋),可能是A、K或G、A極(對單向可控硅)也可能是T2、T1或T2、G極(對雙向可控硅)。若其中有一次測量指示為幾十至幾百歐,則必為單向可控硅。且紅筆所接為K極,黑筆接的為G極,剩下即為A極。若正、反向測批示均為幾十至幾百歐,則必為雙向可控硅。再將旋鈕撥至R×1或R×10擋復測,其中必有一次阻值稍大,則稍大的一次紅筆接的為G極,黑筆所接為T1極,餘下是T2極。
  2、性能的差別:將旋鈕撥至R×1擋,對於1~6A單向可控硅,紅筆接K極,黑筆同時接通G、A極,在保持黑筆不脫離A極狀態下斷開G極,指針應指示幾十歐至一百歐,此時可控硅已被觸發,且觸發電壓低(或觸發電流小)。然後瞬時斷開A極再接通,指針應退回∞位置,則表明可控硅良好。
  對於1~6A雙向可控硅,紅筆接T1極,黑筆同時接G、T2極,在保證黑筆不脫離T2極的前提下斷開G極,指針應指示為幾十至一百多歐(視可控硅電流大小、廠家不同而異)。然後將兩筆對調,重復上述步驟測一次,指針指示還要比上一次稍大十幾至幾十歐,則表明可控硅良好,且觸發電壓(或電流)小。
  若保持接通A極或T2極時斷開G極,指針立即退回∞位置,則說明可控硅觸發電流太大或損壞。可按圖2方法進一步測量,對於單向可控硅,閉合開關K,燈應發亮,斷開K燈仍不息滅,否則說明可控硅損壞。
  對於雙向可控硅,閉合開關K,燈應發亮,斷開K,燈應不息滅。然後將電池反接,重復上述步驟,均應是同一結果,才說明是好的。否則說明該器件已損壞。
