找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 7037|回复: 17
收起左侧

电阻炉温度控制系统设计 详细资料 含仿真 源码 教程

  [复制链接]
ID:129350 发表于 2017-5-7 09:26 | 显示全部楼层 |阅读模式
内含程序,原理图和参考文件

温度设定部分
设定部分主要是键盘输入,此部分主要由三个按键组成,PLAS为加,SUBS为减,START为开始。当系统启动时,默认设定温度为30℃,当按下PLAS时设置水温增加,按下SUBS时设置水温减小,当按下START时为开始加热。
此部分按键为常开按钮,当不触发时为断开状态,按键按下之后与地连接,故为低电平,单片机读取按键数据为低电平时有效。
仿真说明
由于单片机端口的驱动能力有限,所以令其低电平触发光电耦合器,故当P1.5口输出低电平时电热丝加热。
当设定温度与实际温度之差大于10℃时属于粗调,即令电热丝持续加热,无PWM控制;当设定温度与实际温度之差小于10℃时属于微调,即电热丝加热时受PWM控制。
仿真结果
对各温度仿真结果如下:
(1)实际温度19℃,设定温度30℃
当差值大于10℃时,输出为低电平,电热丝持续加热,如图5.1所示:
0.png


(2)实际温度21℃,设定温度30℃当差值小于10℃,控制算法开始起作用,由低电平部分控制加热,如图5.2所示:
0.png

(3)实际温度24℃,设定温度30℃当差值逐渐缩小,低电平时间减少,电热丝功率降低,如图5.3所示:
0.png

(4)实际温度28℃,设定温度30℃当差值逐渐缩小,低电平时间减少,电热丝功率降低,如图5.4所示: 0.png

(5)实际温度32℃,设定温度30℃当设定温度小于实际温度时,PWM输出高电平,电热丝不工作,如图5.5所示:
0.png



1.1 结论(1)当设定温度与实际温度之差小于10℃时,P1.5输出PWM方波控制电热丝加热功率,以达到精确调温的目的。(2)当差值大于10℃时,电热丝持续加热,使其尽快进入到PWM调节阶段(3)当设定温度小于实际温度,说明温度已超调,故电热丝不工作,使其降温。

0.png 0.png

  1. //包含的头文件
  2. #include<REG51.H>                                                                                                                                                  
  3. #include<math.h>
  4. #include<INTRINS.H>
  5. #include<string.h>
  6. //定义宏定义
  7. #define CIRCLE 100
  8. #define uchar unsigned char
  9. #define uint   unsigned int;

  10. struct PID {  
  11. unsigned int SetPoint; // 设定目标 Desired Value  
  12. unsigned int Proportion; // 比例常数 Proportional Const  
  13. unsigned int Integral; // 积分常数 Integral Const  
  14. unsigned int Derivative; // 微分常数 Derivative Const  
  15. unsigned int LastError; // Error[-1]  
  16. unsigned int PrevError; // Error[-2]  
  17. unsigned int SumError; // Sums of Errors  
  18. };  
  19. struct PID spid; // PID Control Structure  
  20. unsigned int rout; // PID Response (Output)  
  21. unsigned int rin; // PID Feedback (Input)  
  22. typedef unsigned char BYTE;
  23. typedef unsigned int WORD;
  24. typedef bit BOOL ;

  25. sbit key1=P3^2;   //定义按键位置
  26. sbit key2=P3^3;
  27. sbit key3=P3^4;

  28. sbit rs = P1^0;
  29. sbit rw = P1^1;
  30. sbit ep = P1^2;
  31. sbit pwm=P1^5; //PWM输出端设置为P1.5输出
  32. sbit DQ=P1^3;//ds18b20 信号引脚即DQ
  33. sfr dataled=0x80;//显示数据端口 即P2口为段选码输入口
  34. uchar temp;//温度变量
  35. unsigned char set_temper=30;  
  36. unsigned char high_time;
  37. unsigned int s;   
  38. uchar flag_get,count,num,counter; //温度读取标志位、中断次数变量t0中断次数以读取温度值, t1中断次数以控制周期和PWM的占空比
  39. char const table[]={100,80,70,60,50,40,30,20,0};//高电平时间查表
  40. uchar
  41. code
  42. tab[]={0x28,0xEB,0x32,0xA2,0xE1,0xA4,0x24,0xEA,0x20,0xA0,0x60,0x25,0x3C,0x23,0x34,
  43. 0x74,0xF7,0xFF};//,0x88,0x83,0xC6,0xA1,0x86,0x8E};//共阳led显示段码115段码表
  44. uchar  shi,ge,danwei,dang,dang_dis;  // 定义温度十位、个位、摄氏度单位、及档位变量名称,及显示档位的查表变量

  45. void delay1(uchar MS);// 延时函数
  46. unsigned char ReadTemperature(void);//读温度子函数
  47. void Init_DS18B20(void);// DS18B20初始化
  48. unsigned char ReadOneChar(void);//读字节子函数
  49. void WriteOneChar(unsigned char dat);//写字节子函数
  50. void delay(unsigned int i);//延时
  51. //void del(unsigned int);
  52. void drive_moto();//由P1低四位输出控制信号
  53. void initial();
  54. unsigned int TempBuffer[5];
  55. BYTE code dis1[] = {"Temperature:"};
  56. BYTE code dis2[] = {"Set_Temper:"};
  57. BYTE code dis3[] = {"0123456789"};
  58. BYTE code dis4[] = 0xdf;
  59. BYTE code dis5[] = {"C"};
  60. //定义函数
  61. void delay2(BYTE ms)
  62. {  
  63. BYTE i;
  64. while(ms--)
  65. {
  66.   for(i = 0; i< 250; i++)
  67.   {
  68.    _nop_();
  69.    _nop_();
  70.    _nop_();
  71.    _nop_();
  72.   }
  73. }
  74. }
  75. // 测试LCD忙碌状态  
  76. BOOL lcd_bz()
  77. {      
  78. BOOL result;
  79. rs = 0;
  80. rw = 1;
  81. ep = 1;
  82. _nop_();
  83. _nop_();
  84. _nop_();
  85. _nop_();
  86. result = (BOOL)(P0 & 0x80);
  87. ep = 0;
  88. return result;  
  89. }
  90. // 写入指令数据到LCD  
  91. void lcd_wcmd(BYTE cmd)
  92. {      
  93. while(lcd_bz());
  94. rs = 0;
  95. rw = 0;
  96. ep = 0;
  97. _nop_();
  98. _nop_();  
  99. P0 = cmd;
  100. _nop_();
  101. _nop_();
  102. _nop_();
  103. _nop_();
  104. ep = 1;
  105. _nop_();
  106. _nop_();
  107. _nop_();
  108. _nop_();
  109. ep = 0;   
  110. }

  111. //设定显示位置
  112. void lcd_pos(BYTE pos)
  113. {      
  114. lcd_wcmd(pos | 0x80);
  115. }
  116. //写入字符显示数据到LCD
  117. void lcd_wdat(BYTE dat)  
  118. {      
  119. while(lcd_bz());
  120. rs = 1;
  121. rw = 0;
  122. ep = 0;
  123. P0 = dat;
  124. _nop_();
  125. _nop_();
  126. _nop_();
  127. _nop_();
  128. ep = 1;
  129. _nop_();
  130. _nop_();
  131. _nop_();
  132. _nop_();
  133. ep = 0;  
  134. }

  135. //LCD初始化设定  
  136. void lcd_init()
  137. {      
  138. lcd_wcmd(0x38);   //
  139. delay2(1);
  140. lcd_wcmd(0x0c);   //
  141. delay2(1);
  142. lcd_wcmd(0x06);   //
  143. delay2(1);
  144. lcd_wcmd(0x01);   //清除LCD的显示内容
  145. delay2(1);
  146. }

  147. //初始化相应的寄存器
  148. void initial()
  149. {
  150. EA=1;
  151. TMOD=0x11;//定时器设置皆工作在16定时计数器模式
  152. TH0=0xef;//T0赋初值
  153. TL0=0xf0;
  154. TH1=(65536-1000)/256;//T1赋初值
  155. TL1=(65536-1000)%256;
  156. ET1=1;
  157. ET0=1;
  158. TR0=1;
  159. TR1=1;
  160. P2=0xff;   
  161. count=0;
  162. counter=0;//pwm占空比控制变量
  163. //IP=0x08; //T1优先级高于T0 ,后来证明此语句是多余的因为当加入此句后反/////而PWM调速和温度检测都变得有点不稳定
  164. }
  165. //延时函数
  166. void delay(unsigned int i)
  167. {
  168. while(i--);  
  169. }

  170. //18b20初始化函数检测总线上是否有从属器件DS的存在若存在则通讯成功
  171. void Init_DS18B20(void)
  172. {
  173. unsigned char x=0;
  174. DQ = 1;    //DQ复位
  175. delay(8);  //稍做延时
  176. DQ = 0;    //单片机将DQ拉低
  177. delay(80); //延时 大于 480us
  178. DQ = 1;    //拉高总线
  179. delay(10);
  180. x=DQ;      //稍做延时后 如果x=0则初始化成功 x=1则初始化失败因为DQ复位成功的回答信号即存在信号是低电平
  181. delay(5);
  182. }

  183. //读一个字节ROM
  184. unsigned char ReadOneChar(void)
  185. {
  186. unsigned char i=0;
  187. unsigned char dat = 0;
  188. for (i=8;i>0;i--)
  189. {
  190.   DQ = 0; // 给脉冲信号
  191.   dat>>=1;
  192.   DQ = 1; // 给脉冲信号
  193.   if(DQ)
  194.    dat|=0x80;//
  195.   delay(5);
  196. }
  197. return(dat);
  198. }

  199. //写一个字节
  200. void WriteOneChar(unsigned char dat)
  201. {
  202. unsigned char i=0;
  203. for (i=8; i>0; i--)
  204. {
  205.   DQ = 0;
  206.   DQ = dat&0x01;
  207.   delay(5);
  208.   DQ = 1;
  209.   dat>>=1;
  210. }
  211. delay(5);
  212. }
  213. //读取温度
  214. unsigned char ReadTemperature(void)
  215. {
  216. unsigned char a=0;
  217. unsigned char b=0;
  218. unsigned char t=0;

  219. Init_DS18B20();// 复位
  220. WriteOneChar(0xCC); // 跳过读序号列号的操作
  221. WriteOneChar(0x44); // 启动温度转换
  222. delay(100);
  223. Init_DS18B20();  //每次操作前都要进行复位
  224. WriteOneChar(0xCC); //跳过读序号列号的操作  
  225. WriteOneChar(0xBE); //读取温度寄存器等共可读9个寄存器 前两个就是温度
  226. a=ReadOneChar();
  227. b=ReadOneChar();
  228. s=(unsigned int)(a&0x0f);  
  229. b<<=4;//取高字节的第四位因为高四位为符号位1111为负0000为正
  230. b+=(a&0xf0)>>4;//忽略小数位取a的高四位与b的相加就是此时所测得到整数温度值
  231. t=b;
  232. return(t);//返回温度值给这个函数
  233. }

  234. //显示函数
  235. void printf()
  236. {
  237. BYTE i;
  238. delay2(10);
  239. lcd_pos(0x00);    // 设置显示位置为第一行的第5个字符
  240. i = 0;
  241. while(dis1[i] != '\0')
  242. {      
  243.   lcd_wdat(dis1[i]);
  244.   i++;
  245. }
  246. lcd_pos(0x40);   // 设置显示位置为第二行第二个字符
  247. i = 0;
  248. while(dis2[i] != '\0')
  249. {
  250.   lcd_wdat(dis2[i]); // 显示字符
  251.   i++;
  252. }
  253. TempBuffer[0]=temp/10;  //十位
  254. TempBuffer[1]=temp%10;  //个位
  255. lcd_pos(0x0c);  
  256. lcd_wdat(dis3[TempBuffer[0]]); //测试温度十位
  257. lcd_pos(0x0d);  
  258. lcd_wdat(dis3[TempBuffer[1]]); //测试温度个位
  259. lcd_pos(0x0e);
  260. lcd_wdat(dis4[0]);  
  261. lcd_pos(0x0f);
  262. lcd_wdat(dis5[0]);

  263. TempBuffer[2]=set_temper/10;  //十位
  264. TempBuffer[3]=set_temper%10;  //个位
  265. lcd_pos(0x4c);  
  266. lcd_wdat(dis3[TempBuffer[2]]); //设置温度十位
  267. lcd_pos(0x4d);  
  268. lcd_wdat(dis3[TempBuffer[3]]); //设置温度个位
  269. lcd_pos(0x4e);
  270. lcd_wdat(dis4[0]);  
  271. lcd_pos(0x4f);
  272. lcd_wdat(dis5[0]);
  273. }

  274. //键盘输入函数
  275. void keyscan()
  276. {
  277.          if(key1==0)
  278.          {
  279.           delay(200);
  280.           if(key1==0)
  281.           {
  282.            set_temper++;
  283.           }
  284.          }
  285.           
  286.          if(key2==0)
  287.          {
  288.           delay(200);
  289.           if(key2==0)
  290.           {
  291.            set_temper--;
  292.           }
  293.          }


  294. //         //开始按键
  295. // if(key3==0)
  296. // {
  297. //          delay(300);
  298. //         if(key3==0)
  299. //         {
  300. //                        TR1=1;       
  301. //         }
  302. // }

  303. }

  304. //T1定时中断服务程序  
  305. void into(void) interrupt 3  
  306. {   
  307.    
  308.     TH1=(65536-1000)/256;//初值重装
  309.     TL1=(65536-1000)%256;
  310. counter++;
  311.   if(counter>CIRCLE)  
  312. {
  313.   counter=0;////限定周期为T=100X0.01ms=10ms
  314. }  
  315. if(counter<=high_time)//取得查表参数以调节pwm的占空比
  316. pwm=1;//PWM高电平
  317. else   
  318. pwm=0;
  319. }


  320. //设定占空比函数
  321. void compare_temper()  
  322. {
  323. char shi=0,ge=0;
  324. if(set_temper>temp)  
  325. {  
  326. //当设置温度与实际温度大于10度时
  327.           if(set_temper-temp>=10)
  328.           {
  329.           //进入死循环
  330.                           while(1)
  331.                           {
  332.                            TR1=0;      //关闭定时器   从而关闭PWM波
  333.                            pwm=0;           //输出低电平
  334.                            temp=ReadTemperature();        //读取温度传感器的值
  335.                            shi=temp/10;                 //分离出十位
  336.                            ge=temp%10;                 //分离出个位
  337.                            lcd_pos(0x0c);         //设置显示位置
  338.                            lcd_wdat(0x30+shi);          //在指定位置显示十位
  339.                            lcd_pos(0x0d);                 //设置显示位置
  340.                            lcd_wdat(0x30+ge);            //在指定位置显示各位

  341.                            if(set_temper-temp<10)         //如果温度差值低于10度
  342.                                            {
  343.                                         TR1=1; //打开定时器  开pwm波
  344.                                            break;         //跳出循环
  345.                                         }
  346.                            }   
  347.                   }  
  348.                   else
  349.                   {
  350.                   //打开定时器
  351.                   TR1=1;
  352.                   //根据差值调整PWM波
  353.                    if (set_temper-temp==9)  
  354.                     high_time=10;
  355.                    if (set_temper-temp==8)  
  356.                     high_time=20;
  357.                    if (set_temper-temp==7)  
  358.                     high_time=30;
  359.                    if (set_temper-temp==6)  
  360.                     high_time=40;
  361.                    if (set_temper-temp==5)  
  362.                     high_time=50;
  363.                    if (set_temper-temp==4)  
  364.                     high_time=60;
  365.                    if (set_temper-temp==3)  
  366.                     high_time=70;
  367.                    if (set_temper-temp==2)  
  368.                     high_time=80;
  369.                    if (set_temper-temp==1)  
  370.                     high_time=90;
  371.                   }  
  372.         }  
  373. else if(set_temper<temp)
  374. {
  375. //关闭定时器
  376.    pwm=1;
  377.    TR1=0;  
  378. }   
  379. }
  380. main()
  381. {
  382. initial();// 初始化个变量及定时器的初值开中断等
  383. lcd_init();    // 初始化LCD  
  384. while(1)
  385. {
  386. //比较函数
  387.   compare_temper();
  388. //输出函数  
  389.   printf();
  390.   //读取温度函数
  391.   

  392. …………限于本文篇幅 余下代码请从51黑下载附件…………

复制代码


0.png

下载:
电阻炉温度控制系统.zip (421.63 KB, 下载次数: 357)

评分

参与人数 2黑币 +10 收起 理由
侯侯 + 5 很给力!
pkh666 + 5 绝世好帖!

查看全部评分

回复

使用道具 举报

ID:89286 发表于 2017-5-28 07:08 | 显示全部楼层
thanks for sharing

评分

参与人数 1黑币 +5 收起 理由
pkh666 + 5 赞一个!

查看全部评分

回复

使用道具 举报

ID:207506 发表于 2017-6-4 14:19 | 显示全部楼层
终于找到我需要的了
回复

使用道具 举报

ID:207506 发表于 2017-6-4 14:46 | 显示全部楼层
thanks for sharing
回复

使用道具 举报

ID:207146 发表于 2017-6-12 21:43 | 显示全部楼层
谢谢分享
回复

使用道具 举报

ID:201547 发表于 2017-6-23 21:45 | 显示全部楼层
谢谢分享
回复

使用道具 举报

ID:263675 发表于 2017-12-28 14:29 | 显示全部楼层
版主附件里代码也不全啊  
回复

使用道具 举报

ID:314339 发表于 2018-4-23 20:24 来自手机 | 显示全部楼层
谢谢楼主,但愿能用
回复

使用道具 举报

ID:338102 发表于 2018-6-1 16:20 来自手机 | 显示全部楼层
感谢分享
回复

使用道具 举报

ID:338102 发表于 2018-6-1 16:21 来自手机 | 显示全部楼层
请问温度控制部分在画电气原理图是电阻丝加热的部分可以加在哪个地方?直接把正玄波信号源换成电阻丝可以吗?
回复

使用道具 举报

ID:323102 发表于 2018-6-7 01:17 | 显示全部楼层
为什么仿真没问题做电路程序就没效果死机怎么办?能不能靠谱点
回复

使用道具 举报

ID:543692 发表于 2019-5-22 18:40 | 显示全部楼层
感谢分享
回复

使用道具 举报

ID:432653 发表于 2020-5-22 20:31 | 显示全部楼层
99度可以做烧开水用
回复

使用道具 举报

ID:89217 发表于 2020-5-25 08:27 | 显示全部楼层
挺不错的文案,双击666
回复

使用道具 举报

ID:903744 发表于 2021-4-16 08:49 来自手机 | 显示全部楼层
有人可以和我讲一下温度控制那部分工作原理吗
回复

使用道具 举报

ID:917567 发表于 2021-6-15 10:46 | 显示全部楼层
请问我的液晶屏为啥不显示呢
回复

使用道具 举报

ID:917567 发表于 2021-6-15 10:49 | 显示全部楼层
请问大家液晶屏都显示么
回复

使用道具 举报

ID:1037640 发表于 2022-7-1 10:01 | 显示全部楼层
为什么仿真运行的时候有logic contentions detected on net #00006呢?求解
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表