news 2026/6/9 14:47:49

基于51单片机设计恒温水箱控制程序仿真加热棒PID算法闭环 采用LCD1602时显示温度值及目标值

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
基于51单片机设计恒温水箱控制程序仿真加热棒PID算法闭环 采用LCD1602时显示温度值及目标值

基于51单片机设计恒温水箱控制程序仿真加热棒PID算法闭环 采用LCD1602时显示温度值及目标值,温度测量范围0~99.9℃,精度±0.1℃; 通过DS18B20温度传感器采集温度作为输入,PID算法控制控制PWM输出,通过继电器控制加热器加热,在温度改变时可以迅速的调整输出; 可通过按键可以设置目标温度值;“功能”键 :按下后切换设置/正常温度控制;“加值”键:在设置时,按下目标温度+1;“减值”键 :在设置时,按下目标温度-1

最近在折腾实验室的恒温水箱控制,发现用51单片机搭个闭环控制系统还挺有意思。今天就跟大伙唠唠这个基于PID算法的温度控制方案,重点说说程序里那些关键代码是怎么落地的。

先说核心部件,DS18B20这玩意儿真是性价比之王。它的单总线协议用51的普通IO口就能驱动,不过时序得抠准了。这里有个读取温度值的代码片段:

float Read_Temperature() { unsigned char LSB, MSB; Init_DS18B20(); Write_DS18B20(0xCC); // 跳过ROM Write_DS18B20(0x44); // 启动转换 Delay(200); // 等转换完成 Init_DS18B20(); Write_DS18B20(0xCC); Write_DS18B20(0xBE); // 读暂存器 LSB = Read_DS18B20(); MSB = Read_DS18B20(); return ((MSB<<8)|LSB)*0.0625; // 转换为摄氏度 }

这里有个坑要注意:DS18B20的转换时间在12位精度时需要750ms,实测发现用200ms延时其实不够稳,后来改用查询Busy位才解决。不过为了代码简洁,示例里还是用了延时方式。

PID控制这块是重头戏,先上结构体定义:

typedef struct { float Kp, Ki, Kd; float Err, LastErr, SumErr; float Output; } PID; void PID_Calc(PID* pid, float current, float target) { pid->Err = target - current; pid->SumErr += pid->Err; // 积分抗饱和 if(pid->SumErr > 200) pid->SumErr = 200; else if(pid->SumErr < -200) pid->SumErr = -200; pid->Output = pid->Kp * pid->Err + pid->Ki * pid->SumErr + pid->Kd * (pid->Err - pid->LastErr); pid->LastErr = pid->Err; }

参数整定是个玄学过程,建议先用Z-N法估算。实测发现对于水箱这种大惯性系统,微分项D可以适当调大,防止过冲。输出量转换成PWM时,记得做限幅处理:

#define PWM_MAX 1000 // 10秒周期 if(pid.Output > PWM_MAX) pid.Output = PWM_MAX; else if(pid.Output <0) pid.Output = 0;

按键处理用状态机实现更靠谱,这里偷懒用查询法:

void Key_Scan() { if(!SET_KEY) { // 功能键按下 mode = !mode; // 切换设置/控制模式 while(!SET_KEY); // 等松手 } if(mode == SET_MODE) { if(!UP_KEY) { target_temp++; Delay(100); } if(!DOWN_KEY) { target_temp--; Delay(100); } // 限制在0-99.9℃ if(target_temp >99.9) target_temp=99.9; if(target_temp <0) target_temp=0; } }

LCD1602显示要注意同时显示当前温度和设定值。用sprintf处理浮点数时,51的内存可能吃紧,建议用自定义函数:

void Show_Temp(float current, float target) { unsigned char buf[16]; // 当前温度 buf[0] = (int)current/10 + '0'; buf[1] = (int)current%10 + '0'; buf[2] = '.'; buf[3] = (int)(current*10)%10 + '0'; buf[4] = '\xDF'; // 摄氏度符号 buf[5] = 'C'; // 目标温度同理... LCD_WriteString(0,0,buf); }

最后说说硬件注意点:继电器的控制引脚记得加反向二极管保护,PWM周期建议10秒左右(加热棒惯性大),测温点尽量靠近加热源。实测下来,这套系统在±0.3℃范围内能稳定控制,比纯开关控制强多了。

完整代码里还有定时器配置、PWM生成这些常规操作,这里不展开了。有坑的地方基本都点到了,剩下的就是耐心调试参数。下次有机会再聊聊怎么用串口做PID参数整定,那又是另一个故事了。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/30 6:20:56

超详细配置说明:lora-scripts中batch_size、lora_rank等参数调优建议

超详细配置说明&#xff1a;lora-scripts中batch_size、lora_rank等参数调优建议 在如今生成式AI快速普及的背景下&#xff0c;越来越多的开发者和创作者希望用自己的数据微调Stable Diffusion或大语言模型&#xff08;LLM&#xff09;&#xff0c;实现风格化输出、IP形象定制甚…

作者头像 李华
网站建设 2026/6/8 12:49:44

如何用lora-scripts自动标注图片prompt?auto_label.py脚本使用详解

如何用 lora-scripts 自动标注图片 prompt&#xff1f;auto_label.py 脚本使用详解 在 AIGC 创作日益普及的今天&#xff0c;越来越多设计师、艺术家和开发者希望训练属于自己的 LoRA 模型——无论是复刻某种艺术风格&#xff0c;还是定制特定角色形象。但一个现实问题是&#…

作者头像 李华
网站建设 2026/5/31 20:05:07

【C++26新特性抢先看】:constexpr变量全面升级,编译期性能提升3倍的秘密

第一章&#xff1a;C26 constexpr变量的演进与意义C 标准的持续演进不断强化编译时计算能力&#xff0c;而 C26 中对 constexpr 变量的进一步扩展标志着这一趋势的重要里程碑。该版本允许更多类型的变量在常量表达式上下文中被求值&#xff0c;显著提升了模板元编程和泛型库的设…

作者头像 李华
网站建设 2026/5/29 7:17:55

C++多线程同步机制全解析(涵盖自旋锁、信号量与futex底层实现)

第一章&#xff1a;C多线程同步机制概述在现代高性能应用程序开发中&#xff0c;多线程编程已成为提升计算效率的关键手段。然而&#xff0c;多个线程并发访问共享资源时&#xff0c;若缺乏有效的同步机制&#xff0c;极易引发数据竞争、状态不一致等问题。C11 标准引入了丰富的…

作者头像 李华
网站建设 2026/5/21 3:38:39

数字人直播带货:24小时不间断的销售终端

数字人直播带货&#xff1a;24小时不间断的销售终端 在电商直播竞争日益白热化的今天&#xff0c;品牌方越来越意识到一个现实问题&#xff1a;真人主播再能说会道&#xff0c;也扛不住每天8小时高强度输出&#xff0c;更别提跨时区全球直播的需求。观众凌晨三点打开直播间&…

作者头像 李华
网站建设 2026/5/10 8:44:22

实时仿真系统效率难题,一文掌握C++物理引擎的高并发处理秘诀

第一章&#xff1a;实时仿真系统效率难题的根源剖析实时仿真系统在工业控制、自动驾驶、航空航天等领域扮演着关键角色&#xff0c;其核心要求是在严格的时间约束下完成计算任务。然而&#xff0c;多数系统在实际运行中面临效率瓶颈&#xff0c;导致响应延迟、资源浪费甚至仿真…

作者头像 李华