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

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

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

PID控制溫度無法控溫怎么判斷哪里出問題?

[復制鏈接]
跳轉到指定樓層
樓主
ID:1125252 發表于 2024-10-14 10:02 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式
PID控制溫度目前是可以加熱,但是到達設定溫度沒法關閉加熱,示波器里顯示時有關閉加熱的部分,不過一下又恢復成加熱了。大部分時間都是加熱。大佬們幫忙分析下是哪里的問題?

#include "include/ca51m020_config.h"               
#include "include/ca51m020sfr.h"
#include "include/ca51m020xsfr.h"
#include "include/gpiodef_m020.h"
#include "include/system_clock.h"

#include "include/uart.h"
#include "include/adc.h"
#include "include/delay.h"


unsigned int count=0;

unsigned int temp_adc=0,cnt=0;
unsigned int PWM_OUT=0;
unsigned int get_ntc=0;
unsigned int AD_Value;
void Timer0_Init(void);
void ADC_Init(void);
unsigned int Get_AdcValue(unsigned char channel);
/*********************************************************************************************************************/

/*********************************************************************************************************************
        ADC控制例程
**********************************************************************************************************************/
/*********************************************************************************************************************/

#define INT_TIME                        1000        //定時時間,單位為us

#define        TH_VAL                                (unsigned char)((0x10000 - (INT_TIME*(FOSC/1000))/12000)>>8)
#define        TL_VAL                                (unsigned char)((0x10000 - (INT_TIME*(FOSC/1000))/12000))

void TIMER0_ISR (void) interrupt 1         //每1ms產生中斷
{
        TH0 = TH_VAL;
        TL0 = TL_VAL;
        count++;
        if(count>=10)                                //10ms
        {
                cnt++;                                //計數
                count=0;
               
        }
      
//        P03 = ~P03;
}

void Timer0_Init(void)
{
        TMOD = (TMOD&0xFC)|0x01;                 //模式選擇: 定時器0,模式1。
        TH0 = TH_VAL;                                    //高8位裝初值
        TL0 = TL_VAL;                                    //低8位裝初值
      
        TR0 = 1;                                               //定時器0使能  
        ET0 = 1;                                               //定時器0中斷使能
      
        EA = 1;
//        PT0 = 1;                                               //設置定時器0中斷優先級為高優先級      
}
/***********************************************************************************
函數名稱: ADC_Init                                                                                       
功能描述: 初始化ADC寄存器(設置ADC時鐘、設置采樣時間、選擇ADC參考電壓)
輸入參數: 無
返 回 值: 無
***********************************************************************************/
void ADC_Init(void)
{
//         ADCON = AST(0) | ADIE(0) | HTME(7) | ADCALE(1) | VSEL(ADC_REF_INNER);        //設置ADC參考電壓為內部1.5V
        ADCON = AST(0) | ADIE(0) | HTME(7) | ADCALE(0) | VSEL(ADC_REF_VDD);                //設置ADC參考電壓為VDD
        ADCFGL = ACKD(7);
        P06F = P06_ADC8_SETTING;                                                                                                //設置P0.6為ADC引腳功能
}

/***********************************************************************************
函數名稱: Get_AdcValue                                                                                       
功能描述: 獲取ADC轉換數值
輸入參數: channel ADC通道號
返 回 值: ADC值
***********************************************************************************/
unsigned int Get_AdcValue(unsigned char channel)
{
        unsigned int AD_Value;
      
        ADCFGL = (ADCFGL&0xE0) | ADCHS(channel);                //選擇ADC通道
        ADCON |= AST(1);                                                                //啟動ADC轉換
        while(!(ADCON & ADIF));                                                        //等待ADC轉換完成
        ADCON |= ADIF;                                                                        //清除ADC中斷標志
        AD_Value = ADCDH*256 + ADCDL;                                        //讀取AD值
        AD_Value >>= 4;               
      
        return AD_Value;
}
/*********************************************************************************************************************/
struct _pid{
        float Set_WD;                        //設定值
        float Actual_WD;                //實際值
      
        float err;                                //偏差
        float err_last;                        //上一次偏差
        float err_next;
        float Kp,Ki,Kd;                       
        float voltage;                        //電壓值
        float integral;                        //積分值
}pid;
void PID_INIT()
{
        pid.Set_WD=0.0;
        pid.Actual_WD=0.0;
        pid.err=0.0;
        pid.err_last=0.0;
        pid.err_next=0.0;
        pid.voltage=0.0;
        pid.integral=0.0;
        pid.Kp=0.3;
        pid.Ki=0.05;
        pid.Kd=0.3;
}
unsigned int temp=40;
void PID_Realize(){
        unsigned int index;
        if(temp==40){temp_adc=1419;}     //設定溫度,對應100K熱敏電阻40攝氏度下的ADC值
        pid.Actual_WD=AD_Value;             //獲取ADC值
        pid.Set_WD=temp_adc;                //設定溫度
        pid.err=pid.Set_WD-pid.Actual_WD;
        if(pid.err>200 || pid.err<-200)
        {
                index=0;
        }
        else if((pid.err<100 && pid.err>0)|| (pid.err<0 && pid.err>-100))
        {
                index=1;
                pid.integral+=pid.err;
        }
        else
        {
                if(pid.err>0)
                {
                        index=(200-pid.err)/100;
                        pid.integral+=pid.err;
                }
                else
                {
                        index=(200+pid.err)/100;
                        pid.integral+=pid.err;
                }
               
        }
        pid.voltage=pid.Kp*pid.err + pid.Ki*pid.integral + pid.Kd*(pid.err-pid.err_last);
      
        pid.err_last=pid.err;
      
      
        PWM_OUT=pid.voltage*1.0;
        if(PWM_OUT>100){PWM_OUT=100;}
}
void hot(unsigned int PWM)
{
        if(cnt<PWM){P07=1;}
        if(cnt>PWM){P07=0;}
        if(cnt>100){cnt=0;}
}
/*********************************************************************************************************************/

void System_Init(void)
{
        LVDCON = 0xE0;                                        //開啟LVD,設置為低電壓復位模式,檢測電壓為2.7V                          
#ifdef SYSCLK_8MHZ                                       
        CKDIV = 0;                                                //系統時鐘上電默認為IRCH的二分頻(4MHz),運行8MHz,則CKDIV設置為0
#endif      
#ifdef UART1_EN
        Uart1_Initial(UART1_BAUTRATE);
#endif      
}
void main(void)
{
      
        P06F = INPUT | PU_EN;                //ADC
        P07F = OUTPUT | PU_EN;                //加熱
        P17F = OUTPUT | PU_EN;
        P17=1;
        P06=1;
        P07=0;                                       
        System_Init();
        EA = 1;                                //開全局中斷
        Timer0_Init();
        ADC_Init();                        //初始化ADC
        PID_INIT();
        while(1)
        {      
                AD_Value = Get_AdcValue(ADC_CH8);        //獲取溫度值
               
                PID_Realize();    //PID控制
                hot(PWM_OUT);    //加熱控制
//                Uart1_PutChar(AD_Value>>8);
//                Uart1_PutChar(AD_Value);
               
               
        }
}


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

使用道具 舉報

沙發
ID:973695 發表于 2024-10-16 15:53 | 只看該作者
在程序中加入動態顯示 PID 控制參數的設定值及 測量到在溫度值,看看邏輯是否正確
回復

使用道具 舉報

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

本版積分規則

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

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 欧美精品网站 | 精品国产成人 | 国产精品久久久久久福利一牛影视 | 四虎永久免费地址 | 国产精品乱码一区二区三区 | 91麻豆精品国产91久久久更新资源速度超快 | 99国内精品久久久久久久 | 毛片入口| av手机免费在线观看 | 国产精品色综合 | 免费不卡视频 | 日本成人午夜影院 | 欧美在线a | 懂色av蜜桃av| 天天操天天拍 | 亚洲天堂久久新 | 久久精品免费 | 久久国产精品久久国产精品 | 久久综合av | 国产在线第一页 | 日韩在线国产 | 欧美在线小视频 | 国产精品无码永久免费888 | 成人av一区二区三区 | 成人亚洲网 | 国产成人精品视频 | 欧美第一区 | a爱视频 | 亚洲一区二区三区在线 | 免费激情av | 成人三级视频 | 少妇精品久久久久久久久久 | 日本电影韩国电影免费观看 | 99精品欧美一区二区三区 | 国产精品免费视频一区 | 亚洲一区中文 | 欧美亚洲激情 | 精品欧美黑人一区二区三区 | 国产一级毛片视频 | 免费视频一区二区 | 99热热99 |