求基于单片机温控的PID算法程序,谢谢啊!!!!

求基于单片机温控的PID算法程序,谢谢啊!!!!,第1张

//温控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=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=0n<timen++)

for(m=0m<2m++){}

}

/***********************************************************

写一位数据子程序

***********************************************************/

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=0i<8i++) /*写一字节数据,一次写一位*/

{

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=0i<2i++){}

value_bit=DQ

EA=1

return(value_bit)

}

/***********************************************************

读一字节数据子程序

***********************************************************/

unsigned char read_byte()

{

unsigned char i,value=0

EA=0

for(i=0i<8i++)

{

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->SumEror // 积分项

+ 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=0i<10i++)

{ 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=0i<10i++)

{ 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=0n<6n++)

{

// k=disp_num1[n]

for(a=0a<8a++)

{

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_()

}

}

以前收藏的一个程序,与你分享一下,希望对你有用、

汇编程序:用减奇数法开平方(16位)

命题:从 1 开始,把连续 n 项奇数的等差数列,求和,可以得到 n^2。

证明:1 + 3 + 5 + ... + (2n-1) = (1 + (2n-1)) * (n/2) = n^2。

那么,对于任意正整数 M,都会有:

M = 1 + 3 + 5 + … + ( 2n - 1 ) + ε

  = n^2 + ε

  = N + ε

式中 N 是完全平方数,N = n^2。

式中 ε 是小于 2n - 1 的误差。

由此,可推出“减奇数开平方”的算法。

即:在 M 中依次减去 1、3、5、...,直到不够减为止;

够减的次数 n,即为 N 的平方根。

程序可见:

这种求平方根的方法,效率很高,远远高于牛顿迭代法。

具体步骤如下:

1、首先,开启我们的keil软件,具体的安装步骤就不做太多的介绍了;

开启后,点击菜单栏上的Project选项,创建我们的工程,如图所示

2、点击后,d出图中的窗口,在箭头所指的文本框中输入你的芯片型号进行查找,如果没有需要自行将芯片添加到keil软件的库下,如图所示;

3、选中芯片点击Ok即可,在左侧的文本框中,选中图中选项右键选择Add New...选项,如图所示;

4、选择.c的文件,输入文件名以及文件的存储目录,如图所示;

5、点击OK后,在我们添加的文本框中输入我们的程序代码,如图所示;

6、编写好代码以后,选择菜单栏上的魔法棒选项,如图所示;

7、选择Output选项,将图中箭头所指的选项,选中,点击ok,如图所示;

8、然后点击编译选项,如图所示;

编译完成后,在我们的文件夹下找到.hex的文件,将其烧写到我们的芯片中即可。


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/yw/11125805.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-13
下一篇 2023-05-13

发表评论

登录后才能评论

评论列表(0条)

保存