久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 2803|回復: 0
打印 上一主題 下一主題
收起左側

單片機PID算法控制電機轉速Proteus仿真程序

[復制鏈接]
跳轉到指定樓層
樓主


單片機源程序如下:
#include<reg52.h>
#include"lcd1602.h"

sfr T2MOD = 0x0c9;
#define uchar unsigned char
#define uint unsigned int

sbit Q0 = P2^4;
sbit Q1 = P2^5;
sbit Q2 = P2^6;
sbit Q3 = P2^7;

sbit PWM                 = P1^7;
sbit UP                         = P1^0;
sbit DOWM                 = P1^1;
sbit GORB                = P2^3; //換相
sbit ADDSPEED         = P1^2;
sbit SUBSPEED        = P1^3;

uint tuint = 65535;
uint tpwm = 1;        //pwm周期為10000us tpwm變量表示pwm高電平時間,也相當于占空比 (仿真時,頻率高時,電機反應慢。在實物上要加大頻率)
uchar t1_flag = 0;

uint pulse = 0;
uint t0_flag = 0;
uchar t2_flag = 0;
bit t2_over = 0;
bit Just_Get = 1;


#define         ZZ                 { Q0 = 0;Q1 = 0;Q2 = 1;Q3 = 1;}        //正轉
#define         FZ                 { Q0 = 1;Q1 = 1;Q2 = 0;Q3 = 0;}        //反轉
#define         STOP        { Q0 = 1;Q1 = 0;Q2 = 1;Q3 = 0;}        //停止
//禁止出現 Q0 = 0;Q1 = 1;Q2 = 0;Q3 = 1; 不然會燒掉mos管

//************************ PID *************************************
float now = 0,bef = 0,bbef = 0;         //本次采樣值,上次采樣值,上上次采樣值
float err_now,err_bef,err_bbef;                //當前偏差,上次偏差,上上次偏差
float error_add = 0;                                //所有偏差之和
float set = 25;                                                //設定值

float kp = 25;
float ki = 25;
float kd = 0;

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

void delayms(uint ms)//延時?個 ms
{
    uchar a,b,c;
        while(ms--)
        {
          for(c=1;c>0;c--)
        for(b=142;b>0;b--)
            for(a=2;a>0;a--);
        }
}

void timer_init()
{
        EA = 1;
        ET0 = 1;
        ET1 = 1;
        ET2 = 1;
        
        TMOD = 0x15; //定時器0 計數模式 定時器1模式1
        T2MOD = 0x01;
        
        TH0 = TL0 = 255;
        TH2 = 0x3C;
        TL2 = 0xB0;                //50MS
        
}
void timer1() interrupt 3
{
        if(t1_flag == 0)
        {
                t1_flag = 1;
                PWM = 1;
                TH1 = (tuint - tpwm + 1)/256;
                TL1 = (tuint - tpwm + 1)%256;
               
        }
        else
        {
                t1_flag = 0;
                PWM = 0;
                TH1 = (tuint - 10000 + tpwm + 1)/256;
                TL1 = (tuint - 10000 + tpwm + 1)%256;
        }
}

void timer0() interrupt 1
{
        TH0 = TL0 = 255;
        t0_flag++;
}
void timer2() interrupt 5
{
        TF2 = 0;
        TH2 = 0x3C;
        TL2 = 0xB0;                //50MS
        
        t2_flag++;
        
        if(t2_flag == 2)
        {
                TR0 = 0;
                TR2 = 0;
                t2_flag = 0;
                t2_over = 1;        //表示100ms時間到
        }
}
void GetPulse()
{
        t0_flag = 0;
        t2_flag = 0;
        
        TH0 = TL0 = 255;
        TH2 = 0x3C;
        TL2 = 0xB0;                //50MS
        
        TR0 = 1;
        TR2 = 1;
}

int PID()        //增量式PID
{
        int change;

        err_now = set - now;
        err_bef = set - bef;
        err_bbef = set - bbef;
        
        change = kp*(err_now - err_bef) + ki*err_now + kd*(err_now - 2*err_bef + err_bbef);
        
/*        
        if(set >= now)
        {        
                if(set - now > 1)
                        change = kp*(err_now - err_bef) + ki*err_now + kd*(err_now - 2*err_bef + err_bbef);
                else
                        change = 0.2*kp*(err_now - err_bef) + 0.5*ki*err_now + kd*(err_now - 2*err_bef + err_bbef);
        }
        else if(now > set)
        {
                if(now - set > 1)
                        change = kp*(err_now - err_bef) + ki*err_now + kd*(err_now - 2*err_bef + err_bbef);
                else
                        change = 0.2*kp*(err_now - err_bef) + 0.5*ki*err_now + kd*(err_now - 2*err_bef + err_bbef);
                        
        }
*/
        
        //change = (kp + ki + kd)*(set - now) + (-kp - 2*kd)*(set - bef) + kd*(set - bbef);
        //change = kp*(set - now) + ki*(set - bef) + kd*(set - bbef);
        if(change > 0)
        {
                printchar(1,10,'+');        
                printuint(1,11,4,change);
               
        }
        else if(change < 0)
        {        
                printchar(1,10,'-');
                printuint(1,11,4,-change);
        }
        else if(change == 0)
        {        
                printchar(1,10,' ');
                printword(1,11," 0  ");

        }
        
        return(change);
}

int PID2()                //位置式PID
{
        
        int num = 0;
        static num_bef = 0;
        
        err_now = set - now;
        err_bef = set - bef;
        
        error_add = error_add + err_now;  //誤差累加。一旦誤差為0則error_add的值不變,PID輸出值不變

        num = kp*err_now + ki*error_add + kd*(err_now - err_bef);
        
/*        
        if(set - now >= 0)
        {        
                if(set - now > 1)
                        num = kp*err_now + ki*error_add + kd*(err_now - err_bef);
                else
                        num = 0.1*kp*err_now + ki*error_add + kd*(err_now - err_bef);
        }
        else
        {
                if(now - set > 1)
                        num = kp*err_now + ki*error_add + kd*(err_now - err_bef);
                else
                        num = 0.1*kp*err_now + ki*error_add + kd*(err_now - err_bef);
                        
        }
        */
        
        if(num > num_bef)
        {
                printchar(1,10,'+');        
                printuint(1,11,4,num - num_bef);
        }
        else if(num < num_bef)
        {
                printchar(1,10,'-');        
                printuint(1,11,4,num_bef - num);
        }
        else
        {        
                printchar(1,10,' ');
                printuint(1,11,4,0);
        }
        
        num_bef = num;
        
        return((uint)num);
}

void main()
{        
        
        lcd_init();
        timer_init();
        TH1 = TL1 = 255;
        
        printword(0,0,"P:");                //比例系數
        printword(0,5,"S:");                //設定值
        printword(1,0,"TPWM:");                //當前占空比
        printword(0,10,"PS:");                //當前電機反饋的每秒脈沖數
        
        while(1)
        {
                if(GORB == 1)
                {        ZZ;                }
                else
                {        FZ;                }
               
                if(ADDSPEED == 0)
                        set++;
                if(SUBSPEED == 0)
                        set--;
               
                if(Just_Get == 1)
                {        
                        Just_Get = 0;
                        GetPulse();
                }
                else if(t2_over == 1)
                {        
                        t2_over = 0;
                        Just_Get = 1;
                        pulse = t0_flag;
                        bbef = bef;
                        bef = now;
                        now = t0_flag;
                        
                        if(set != 0)
                        {
                                TR1 = 1;
                        }
                        else
                        {
                                TR1 = 0;
                                PWM = 0;
                        }
                        
                //        tpwm = tpwm + PID();                //增量式PID
                        tpwm = PID2();                                //位置式PID
                                       
                }
               
                if(UP == 0)
                        kp = kp + 1;
                if(DOWM == 0)
                        kp = kp - 1;
               
                printuint(0,2,3,kp);
                printuint(0,7,3,set);
                printuint(1,5,4,tpwm);
                printuint(0,13,5,pulse);

        }
        
}
        

2.jpg (20.8 KB, 下載次數: 117)

2.jpg

PID算法控制電機轉速.rar

100.22 KB, 下載次數: 42, 下載積分: 黑幣 -5

評分

參與人數 1黑幣 +50 收起 理由
admin + 50 共享資料的黑幣獎勵!

查看全部評分

分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏2 分享淘帖 頂 踩
回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

手機版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 在线日韩欧美 | 超碰成人免费观看 | 黄色精品 | 亚洲视频免费在线观看 | 97超碰在线播放 | 91高清免费 | 成人欧美一区二区三区1314 | av资源网站 | 久久综合成人精品亚洲另类欧美 | 999久久久久久久久6666 | 毛片免费看的 | 中文在线一区二区 | 国产三区在线观看视频 | 欧美日韩精品一区二区三区四区 | 亚洲国产中文在线 | 国产精品日韩欧美一区二区 | 婷婷久久精品一区二区 | 亚洲第一网站 | 美女一级a毛片免费观看97 | 我我色综合 | 精品网| 精品三级在线观看 | 久久一区二区三区四区 | 国产在线精品一区二区 | 精品伊人 | 欧美国产日韩精品 | 精品国产欧美一区二区三区成人 | 日韩欧美中文字幕在线观看 | 久久久国产一区二区三区 | 中文字幕一区二区三区四区 | 成人免费视频网站在线观看 | 成人综合一区二区 | 9191av| 婷婷丁香综合网 | 午夜激情视频 | 欧洲高清转码区一二区 | 人人做人人澡人人爽欧美 | 成人av在线播放 | 亚洲国产一区二区三区在线观看 | 午夜精品一区二区三区在线观看 | 国产精品区二区三区日本 |