|
單片機源程序:
- #include<REG51.H>
- #include<math.h>
- #include<INTRINS.H>
- #include<string.h>
- #define CIRCLE 100
- #define uchar unsigned char
- #define uint unsigned int;
- 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)
- typedef unsigned char BYTE;
- typedef unsigned int WORD;
- typedef bit BOOL ;
- sbit key1=P3^2; //定義按鍵位置
- sbit key2=P3^3;
- sbit rs = P1^0;
- sbit rw = P1^1;
- sbit ep = P1^2;
- sbit pwm=P1^5; //PWM輸出端設置為P1.5輸出
- sbit DQ=P1^3;//ds18b20 信號引腳即DQ
- sfr dataled=0x80;//顯示數據端口 即P2口為段選碼輸入口
- uchar temp;//溫度變量
- unsigned char set_temper=30;
- unsigned char high_time;
- unsigned int s;
- uchar flag_get,count,num,counter; //溫度讀取標志位、中斷次數變量t0中斷次數以讀取溫度值, t1中斷次數以控制周期和PWM的占空比
- char const table[]={100,80,70,60,50,40,30,20,0};//高電平時間查表
- uchar
- code
- tab[]={0x28,0xEB,0x32,0xA2,0xE1,0xA4,0x24,0xEA,0x20,0xA0,0x60,0x25,0x3C,0x23,0x34,
- 0x74,0xF7,0xFF};//,0x88,0x83,0xC6,0xA1,0x86,0x8E};//共陽led顯示段碼115段碼表
- uchar shi,ge,danwei,dang,dang_dis; // 定義溫度十位、個位、攝氏度單位、及檔位變量名稱,及顯示檔位的查表變量
-
- void delay1(uchar MS);// 延時函數
- unsigned char ReadTemperature(void);//讀溫度子函數
- void Init_DS18B20(void);// DS18B20初始化
- unsigned char ReadOneChar(void);//讀字節子函數
- void WriteOneChar(unsigned char dat);//寫字節子函數
- void delay(unsigned int i);//延時
- //void del(unsigned int);
- void drive_moto();//由P1低四位輸出控制信號
- void initial();
- unsigned int TempBuffer[5];
- BYTE code dis1[] = {"Temperature:"};
- BYTE code dis2[] = {"Set_Temper:"};
- BYTE code dis3[] = {"0123456789"};
- BYTE code dis4[] = 0xdf;
- BYTE code dis5[] = {"C"};
- //
- void init();
- void delay2(BYTE ms)
- {
- BYTE i;
- while(ms--)
- {
- for(i = 0; i< 250; i++)
- {
- _nop_();
- _nop_();
- _nop_();
- _nop_();
- }
- }
- }
-
- BOOL lcd_bz()
- { // 測試LCD忙碌狀態
- BOOL result;
- rs = 0;
- rw = 1;
- ep = 1;
- _nop_();
- _nop_();
- _nop_();
- _nop_();
- result = (BOOL)(P0 & 0x80);
- ep = 0;
- return result;
- }
-
- void lcd_wcmd(BYTE cmd)
- { // 寫入指令數據到LCD
- while(lcd_bz());
- rs = 0;
- rw = 0;
- ep = 0;
- _nop_();
- _nop_();
- P0 = cmd;
- _nop_();
- _nop_();
- _nop_();
- _nop_();
- ep = 1;
- _nop_();
- _nop_();
- _nop_();
- _nop_();
- ep = 0;
- }
- void lcd_pos(BYTE pos)
- { //設定顯示位置
- lcd_wcmd(pos | 0x80);
- }
- void lcd_wdat(BYTE dat)
- { //寫入字符顯示數據到LCD
- while(lcd_bz());
- rs = 1;
- rw = 0;
- ep = 0;
- P0 = dat;
- _nop_();
- _nop_();
- _nop_();
- _nop_();
- ep = 1;
- _nop_();
- _nop_();
- _nop_();
- _nop_();
- ep = 0;
- }
- void lcd_init()
- { //LCD初始化設定
- lcd_wcmd(0x38); //
- delay2(1);
- lcd_wcmd(0x0c); //
- delay2(1);
- lcd_wcmd(0x06); //
- delay2(1);
- lcd_wcmd(0x01); //清除LCD的顯示內容
- delay2(1);
- }
- void initial()
- {
- EA=1;
- TMOD=0x11;//定時器設置皆工作在16定時計數器模式
- TH0=0xef;//T0賦初值
- TL0=0xf0;
- TH1=(65536-1000)/256;//T1賦初值
- TL1=(65536-1000)%256;
- ET1=1;
- ET0=1;
- TR0=1;
- TR1=1;
- P2=0xff;
- count=0;
- counter=0;//pwm占空比控制變量
- //IP=0x08; //T1優先級高于T0 ,后來證明此語句是多余的因為當加入此句后反/////而PWM調速和溫度檢測都變得有點不穩定
- }
- void delay(unsigned int i)//延時函數
- {
- while(i--);
- }
- //18b20初始化函數檢測總線上是否有從屬器件DS的存在若存在則通訊成功
- void Init_DS18B20(void)
- {
- unsigned char x=0;
- DQ = 1; //DQ復位
- delay(8); //稍做延時
- DQ = 0; //單片機將DQ拉低
- delay(80); //延時 大于 480us
- DQ = 1; //拉高總線
- delay(10);
- x=DQ; //稍做延時后 如果x=0則初始化成功 x=1則初始化失敗因為DQ復位成功的回答信號即存在信號是低電平
- delay(5);
- }
- //讀一個字節ROM
- unsigned char ReadOneChar(void)
- {
- unsigned char i=0;
- unsigned char dat = 0;
- for (i=8;i>0;i--)
- {
- DQ = 0; // 給脈沖信號
- dat>>=1;
- DQ = 1; // 給脈沖信號
- if(DQ)
- dat|=0x80;//
- delay(5);
- }
- return(dat);
- }
-
- //寫一個字節
- void WriteOneChar(unsigned char dat)
- {
- unsigned char i=0;
- for (i=8; i>0; i--)
- {
- DQ = 0;
- DQ = dat&0x01;
- delay(5);
- DQ = 1;
- dat>>=1;
- }
- delay(5);
- }
- //讀取溫度
- unsigned char ReadTemperature(void)
- {
- unsigned char a=0;
- unsigned char b=0;
- unsigned char t=0;
-
- Init_DS18B20();// 復位
- WriteOneChar(0xCC); // 跳過讀序號列號的操作
- WriteOneChar(0x44); // 啟動溫度轉換
- delay(100);
- Init_DS18B20(); //每次操作前都要進行復位
- WriteOneChar(0xCC); //跳過讀序號列號的操作
- WriteOneChar(0xBE); //讀取溫度寄存器等共可讀9個寄存器 前兩個就是溫度
- a=ReadOneChar();
- b=ReadOneChar();
- s=(unsigned int)(a&0x0f);
- b<<=4;//取高字節的第四位因為高四位為符號位1111為負0000為正
- b+=(a&0xf0)>>4;//忽略小數位取a的高四位與b的相加就是此時所測得到整數溫度值
- t=b;
- return(t);//返回溫度值給這個函數
- }
-
- void printf()
- {
- BYTE i;
- delay2(10);
- lcd_pos(0x00); // 設置顯示位置為第一行的第5個字符
- i = 0;
- while(dis1[i] != '\0')
- {
- lcd_wdat(dis1[i]);
- i++;
- }
- lcd_pos(0x40); // 設置顯示位置為第二行第二個字符
- i = 0;
- while(dis2[i] != '\0')
- {
- lcd_wdat(dis2[i]); // 顯示字符
- i++;
- }
- TempBuffer[0]=temp/10; //十位
- TempBuffer[1]=temp%10; //個位
- lcd_pos(0x0c);
- lcd_wdat(dis3[TempBuffer[0]]); //測試溫度十位
- lcd_pos(0x0d);
- lcd_wdat(dis3[TempBuffer[1]]); //測試溫度個位
- lcd_pos(0x0e);
- lcd_wdat(dis4[0]);
- lcd_pos(0x0f);
- lcd_wdat(dis5[0]);
-
- TempBuffer[2]=set_temper/10; //十位
- TempBuffer[3]=set_temper%10; //個位
- lcd_pos(0x4c);
- lcd_wdat(dis3[TempBuffer[2]]); //設置溫度十位
- lcd_pos(0x4d);
- lcd_wdat(dis3[TempBuffer[3]]); //設置溫度個位
- lcd_pos(0x4e);
- lcd_wdat(dis4[0]);
- lcd_pos(0x4f);
- lcd_wdat(dis5[0]);
- }
- void keyscan()
- {
- if(key1==0)
- {
- delay(200);
- if(key1==0)
- {
- set_temper++;
- }
- }
- if(key2==0)
- {
- delay(200);
- if(key2==0)
- {
- set_temper--;
- }
- }
- }
- void tim(void) interrupt 1 using 1//中斷用于數碼管掃描和溫度檢測間隔
- {
- // TH0=0xef;//定時器重裝值
- // TL0=0xf0;
- // num++;
- // if (num==100)
- // {
- // num=0;
- // flag_get=1;//標志位有效
- // }
- }
-
- void into(void) interrupt 3 //T1定時中斷服務程序
- {
-
- TH1=(65536-1000)/256;//初值重裝
- TL1=(65536-1000)%256;
- counter++;
- if(counter>CIRCLE)
- {
- counter=0;////限定周期為T=100X0.01ms=10ms
- }
- if(counter<=high_time)//取得查表參數以調節pwm的占空比
- pwm=1;//PWM高電平
- else
- pwm=0;
- }
-
- void compare_temper()
- {
- // unsigned char i;
- if(set_temper>temp)
- {
- if(set_temper-temp>=10)
- {
- high_time=100;
- }
- else
- {
- // for(i=0;i<10;i++)
- // {
- // rin = s; // Read Input
- // rout = PIDCalc ( &spid,rin ); // Perform PID Interation
- // }
- if (set_temper-temp==9)
- high_time=90;
- if (set_temper-temp==8)
- high_time=80;
- if (set_temper-temp==7)
- high_time=70;
- if (set_temper-temp==6)
- high_time=60;
- if (set_temper-temp==5)
- high_time=50;
- if (set_temper-temp==4)
- high_time=40;
- if (set_temper-temp==3)
- high_time=30;
- if (set_temper-temp==2)
- high_time=20;
- if (set_temper-temp==1)
- high_time=10;
- }
- }
- else if(set_temper<temp)
- {
- // if(temp-set_temper>0)
- // {
- high_time=0;
- // }
- }
- }
- main()
- {
- initial();// 初始化個變量及定時器的初值開中斷等
- lcd_init(); // 初始化LCD
- while(1)
- {
- compare_temper();
- printf();
- temp=ReadTemperature();
- keyscan();
- }
- }
復制代碼
所有資料打包下載:
爐溫控制系統.rar
(382.23 KB, 下載次數: 263)
2016-11-30 11:10 上傳
點擊文件名下載附件
|
評分
-
查看全部評分
|