用51单片机和LD3320语音模块DIY一个会说话的温度报警器(附完整代码)
在电子DIY的世界里,将冰冷的硬件赋予"说话"的能力总是令人兴奋。想象一下,当环境温度超过设定值时,设备不仅能发出警报,还能用语音提醒"温度过高,请注意降温",这种交互体验远比单调的蜂鸣声更有温度。本文将带你用经典的STC89C52单片机、LD3320语音识别模块和DS18B20温度传感器,打造这样一个会"说话"的智能温度报警器。
1. 硬件选型与电路设计
1.1 核心器件选型对比
| 器件类型 | 候选方案 | 最终选择 | 选择理由 |
|---|---|---|---|
| 主控MCU | STM32F103C8T6 | STC89C52RC | 成本更低,51架构更易上手,资源完全满足需求 |
| 语音模块 | SYN7318 | LD3320 | 本地识别无需联网,支持自定义词条,响应速度更快 |
| 温度传感器 | DHT11 | DS18B20 | 测量范围更广(-55~125℃),精度±0.5℃,单总线协议节省IO口 |
| 报警输出 | 无源蜂鸣器 | 有源蜂鸣器+语音 | 双重报警机制,语音提示更直观 |
1.2 关键电路连接要点
最小系统电路:
- 晶振电路:11.0592MHz晶振+30pF电容×2
- 复位电路:10kΩ电阻+10μF电容构成上电复位
- 电源滤波:VCC与GND间加0.1μF去耦电容
LD3320语音模块接线:
P1.1 → LD3320的GPIO1 // 识别状态检测 P3.0 → LD3320的TXD // 串口接收 P3.1 ← LD3320的RXD // 串口发送 +5V → VCC GND → GNDDS18B20典型接法:
VCC ——┬—— 4.7kΩ上拉电阻 │ DS18B20 DATA引脚 —— P2.0 │ GND ——┘注意:DS18B20的数据线必须加上拉电阻,否则无法正常通信。长距离传输时可降低上拉电阻值至2.2kΩ。
2. 软件开发环境搭建
2.1 Keil C51工程配置
新建工程时选择"AT89C52"作为目标器件
在Options for Target中设置:
- Target标签页:勾选"Use On-chip ROM"
- Output标签页:勾选"Create HEX File"
- C51标签页:优化等级设为8级优化
添加必要的头文件路径:
- DS18B20驱动头文件
- LCD1602显示驱动
- LD3320控制库
2.2 语音词条配置技巧
使用LD3320配套的上位机软件配置识别词条时:
- 将"kai ji bao jing"映射到0x01指令码
- "guan bi bao jing"映射到0x02
- "wen du bao jing"映射到0x03
- 每个词条后添加0xFF作为结束符
// 典型语音指令处理逻辑 if(voice_cmd == 0x01) { alarm_enable = 1; LD3320_PlayMP3(1); // 播放"报警已开启"语音 } else if(voice_cmd == 0x02) { alarm_enable = 0; LD3320_PlayMP3(2); // 播放"报警已关闭"语音 }3. 核心功能实现
3.1 温度采集与处理
DS18B20的驱动关键在于精确的时序控制,以下是温度读取的核心函数:
float Read_Temperature(void) { unsigned char LSB, MSB; int temp; float f_temp; DS18B20_Reset(); // 复位 DS18B20_WriteByte(0xCC); // 跳过ROM DS18B20_WriteByte(0x44); // 启动转换 Delay_ms(750); // 等待转换完成 DS18B20_Reset(); DS18B20_WriteByte(0xCC); // 跳过ROM DS18B20_WriteByte(0xBE); // 读取暂存器 LSB = DS18B20_ReadByte(); // 读取低字节 MSB = DS18B20_ReadByte(); // 读取高字节 temp = (MSB << 8) | LSB; f_temp = temp * 0.0625; // 转换为实际温度 return f_temp; }3.2 语音报警逻辑设计
实现分级语音报警功能:
- 当温度超过阈值5℃以内,播放"温度偏高"提示
- 超过5~10℃,播放"温度过高,请及时处理"
- 超过10℃以上,循环播放"危险!高温警报!"
void Voice_Alarm(float current, float threshold) { float delta = current - threshold; if(delta > 0 && delta <= 5) { LD3320_PlayMP3(3); // 温和提醒 } else if(delta > 5 && delta <=10) { LD3320_PlayMP3(4); // 强烈提醒 Beep_On(1000); // 1kHz蜂鸣 } else if(delta > 10) { while(1) { // 紧急状态循环报警 LD3320_PlayMP3(5); Beep_On(2000); // 2kHz高频蜂鸣 Delay_ms(500); Beep_Off(); Delay_ms(500); } } }4. 系统调试与优化
4.1 常见问题排查指南
| 现象 | 可能原因 | 解决方案 |
|---|---|---|
| LD3320无反应 | 供电不足 | 检查5V电源电流是否≥500mA |
| 波特率不匹配 | 确认模块与单片机波特率均为9600 | |
| DS18B20读数异常 | 上拉电阻缺失 | 在DATA线添加4.7kΩ上拉 |
| 时序不符合要求 | 用示波器检查复位脉冲是否达到480μs | |
| 语音识别率低 | 环境噪音干扰 | 增加硅麦的指向性,远离风扇等噪音源 |
| 词条设计不合理 | 避免相似发音词条,如"开始"和"考试" |
4.2 性能优化技巧
降低功耗:
- 在温度稳定时让单片机进入空闲模式
- 使用
PCON |= 0x01进入IDLE模式 - 通过外部中断唤醒
提高响应速度:
// 优化后的主循环结构 void main() { System_Init(); while(1) { if(need_temp_update) { current_temp = Read_Temperature(); Update_Display(); Check_Alarm(); need_temp_update = 0; } Idle_Mode(); // 进入低功耗状态 } }- 增强语音识别:
- 在LD3320的GPIO引脚添加LED指示
- 识别时点亮,方便观察状态
- 调整MIC增益电位器至最佳灵敏度
5. 项目扩展方向
5.1 添加无线传输功能
通过ESP-01S WiFi模块将温度数据上传至物联网平台:
void ESP8266_SendData(float temp) { UART_SendString("AT+CIPSTART=\"TCP\",\"api.thingspeak.com\",80\r\n"); Delay_ms(1000); char cmd[128]; sprintf(cmd, "GET /update?api_key=YOUR_KEY&field1=%.2f\r\n", temp); UART_SendString("AT+CIPSEND="); UART_SendNumber(strlen(cmd)); UART_SendString("\r\n"); Delay_ms(500); UART_SendString(cmd); }5.2 多传感器融合方案
结合DHT11实现温湿度综合监测:
在PCB布局时:
- 将DS18B20远离MCU放置,避免自热影响
- DHT11应避开直接风吹位置
数据融合算法:
float Calc_DiscomfortIndex(float T, float RH) { // 计算温湿指数 return T - 0.55*(1 - 0.01*RH)*(T - 14.5); }5.3 外壳设计与声学优化
3D打印外壳注意事项:
- 为蜂鸣器预留共振腔
- MIC开孔直径2-3mm最佳
- 散热孔布置在传感器附近
提升语音质量:
- 在LD3320的音频输出端添加LM386功放电路
- 选用8Ω/1W的喇叭
- 箱体内部填充吸音棉减少回声
附录:完整工程代码
#include <reg52.h> #include <intrins.h> // 硬件引脚定义 sbit DS18B20_DQ = P2^0; sbit BEEP = P1^5; sbit LD3320_STA = P1^1; // 全局变量 float temperature = 0; float threshold = 30.0; // 默认阈值 // DS18B20延时函数 void Delay_us(unsigned int us) { while(us--) { _nop_(); _nop_(); _nop_(); _nop_(); } } // DS18B20初始化 bit DS18B20_Init() { bit ack; DS18B20_DQ = 1; Delay_us(5); DS18B20_DQ = 0; Delay_us(500); DS18B20_DQ = 1; Delay_us(60); ack = DS18B20_DQ; Delay_us(240); return ack; } // 读取一个字节 unsigned char DS18B20_ReadByte() { unsigned char i, dat = 0; for(i=0; i<8; i++) { DS18B20_DQ = 0; _nop_(); _nop_(); dat >>= 1; DS18B20_DQ = 1; _nop_(); _nop_(); if(DS18B20_DQ) dat |= 0x80; Delay_us(60); } return dat; } // 写入一个字节 void DS18B20_WriteByte(unsigned char dat) { unsigned char i; for(i=0; i<8; i++) { DS18B20_DQ = 0; _nop_(); _nop_(); DS18B20_DQ = dat & 0x01; Delay_us(60); DS18B20_DQ = 1; dat >>= 1; } } // 获取温度值 float Get_Temperature() { unsigned char LSB, MSB; int temp; DS18B20_Init(); DS18B20_WriteByte(0xCC); // 跳过ROM DS18B20_WriteByte(0x44); // 启动转换 Delay_ms(750); DS18B20_Init(); DS18B20_WriteByte(0xCC); // 跳过ROM DS18B20_WriteByte(0xBE); // 读取暂存器 LSB = DS18B20_ReadByte(); MSB = DS18B20_ReadByte(); temp = (MSB << 8) | LSB; return temp * 0.0625; } // 语音报警函数 void Voice_Alert(float temp) { if(temp > threshold) { BEEP = 1; // 开启蜂鸣器 if(LD3320_STA == 1) { // 语音模块空闲 if(temp - threshold > 10) { UART_SendString("play 3\r\n"); // 紧急语音 } else { UART_SendString("play 2\r\n"); // 普通警报 } } } else { BEEP = 0; // 关闭蜂鸣器 } } void main() { UART_Init(); // 串口初始化 while(1) { temperature = Get_Temperature(); Voice_Alert(temperature); Delay_ms(1000); // 1秒更新一次 } }这个项目最有趣的部分是当语音模块突然说出"检测到高温"时,总能引起周围人的会心一笑。在实际测试中,将DS18B20靠近笔记本电脑的散热口,听到它用焦急的语气报警的那一刻,所有的调试辛苦都变得值得。